再来点boss题解

再来点boss

给定一个 n 个点 m 条边的无向图,保证图连通。找到两个点 s,t ,使得 s 到 t 必须经过的边最多(一条边无论走哪条路线这一条边都经过,这条边就是必须经过的边)。

首先我们要知道知道同一个边双连通分量中,任意两点之间存在至少两条无重边的简单路径。

我们可以发现同一个边双内的点之间没有必须经过的边。所以在这一道题目中,我们只需要把给我们的图给缩点缩成一棵树,然后跑两边 dfs 求树的直径即可,也就是 2 个板子套一下就可以了。

复制代码
#include<bits/stdc++.h>
using namespace std;
const int N=1200100;
int n,m,idx=1,cnt,in1,in2,num,p,q;
int to[N],nxt[N],head[N],rhead[N];
int dfn[N],low[N],bridge[N],rid[N];
int dis[N];
void add(int h[],int u,int v)
{
    idx++;
    to[idx]=v;
    nxt[idx]=h[u];
    h[u]=idx;
}
void tarjan(int s,int last)//缩点板子,不必多说
{
    dfn[s]=low[s]=++cnt;
    for(int i=head[s];i;i=nxt[i])
    {
        int v=to[i];
        if(i==(last^1)) 
            continue;
        if(!dfn[v])
        {
            tarjan(v,i);
            low[s]=min(low[s],low[v]);
            if(dfn[s]<=low[v])//直接就是一座桥
            {
                bridge[i]=bridge[i^1]=1;
            }
        }
        else 
            low[s]=min(low[s],dfn[v]);
    }
}
void dfs_0(int s)//跑2遍dfs,求树的直径
{
    rid[s]=num;
    for(int i=head[s];i;i=nxt[i])
    {
        int v=to[i];
        if(bridge[i]||rid[v]) 
            continue;
        dfs_0(v);
    }
}
void dfs_1(int s,int fa,int &u)
{
    for(int i=rhead[s];i;i=nxt[i])
    {
        int v=to[i];
        if(v==fa) 
            continue;
        dis[v]=dis[s]+1;
        if(dis[0]<dis[v])
        {
            dis[0]=dis[v];
            u=v;
        }
        dfs_1(v,s,u);
    }
}
int main()
{
    scanf("%d%d",&n,&m);
    for(int i=1;i<=m;i++)
    {
        scanf("%d%d",&in1,&in2);
        add(head,in1,in2);
        add(head,in2,in1);
    }
    tarjan(1,-1);
    for(int i=1;i<=n;i++)
    {
        if(!rid[i])
        {
            num++;
            dfs_0(i);
        }
    }
    for(int s=1;s<=n;s++)
    {
        for(int i=head[s];i;i=nxt[i])
        {
            if(rid[s]!=rid[to[i]])
            {
                add(rhead,rid[s],rid[to[i]]);
            }
        }
    }
    dis[1]=0;
    dfs_1(1,0,p);
    dis[0]=dis[p]=0;
    dfs_1(p,0,q);
    cout<<dis[0]<<endl;
    return 0;
}
相关推荐
艾莉丝努力练剑38 分钟前
【LeetCode&数据结构】单链表的应用——反转链表问题、链表的中间节点问题详解
c语言·开发语言·数据结构·学习·算法·leetcode·链表
_殊途2 小时前
《Java HashMap底层原理全解析(源码+性能+面试)》
java·数据结构·算法
珊瑚里的鱼6 小时前
LeetCode 692题解 | 前K个高频单词
开发语言·c++·算法·leetcode·职场和发展·学习方法
秋说7 小时前
【PTA数据结构 | C语言版】顺序队列的3个操作
c语言·数据结构·算法
lifallen7 小时前
Kafka 时间轮深度解析:如何O(1)处理定时任务
java·数据结构·分布式·后端·算法·kafka
liupenglove7 小时前
自动驾驶数据仓库:时间片合并算法。
大数据·数据仓库·算法·elasticsearch·自动驾驶
python_tty8 小时前
排序算法(二):插入排序
算法·排序算法
然我8 小时前
面试官:如何判断元素是否出现过?我:三种哈希方法任你选
前端·javascript·算法
F_D_Z9 小时前
【EM算法】三硬币模型
算法·机器学习·概率论·em算法·极大似然估计
秋说9 小时前
【PTA数据结构 | C语言版】字符串插入操作(不限长)
c语言·数据结构·算法