hdu4587TWO NODES【割点】

152 篇文章 0 订阅

题意:

询问删除连个点后,最多的连通分量个数

思路:

模板题。。先枚举一个点要删除掉的,再查询在删除这个x点的图上,所有的连通分量,以及删除i点能增加的连通分量

#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<vector>

using namespace std;

const int maxn=5050;    //顶点数
int n,m;//n个点 m条边 顶点下标0~n-1
int dfs_clock;//时钟,每访问一个节点增1
vector<int> G[maxn];//G[i]表示i节点邻接的所有节点
int pre[maxn];//pre[i]表示i节点被第一次访问到的时间戳,若pre[i]==0表示i还未被访问
int low[maxn];//low[i]表示i节点及其后代能通过反向边连回的最早的祖先的pre值
bool iscut[maxn];//标记i节点是不是一个割点
int cut[maxn];//cut[i]表示割i点的时图中联通分量的增加量
vector<pair<int, int> >Bridge;
int ans;

int dfs(int u,int fa,int stop)//求出以u为根节点(u在DFS树中的父节点是fa)的树的所有割顶和桥
{
    if (fa == -1) cut[u]--; //如果是根的话,割时增加的量为“连出去的量减一”
    int lowu=pre[u]=++dfs_clock;
    int child=0;    //子节点数目
    for(int i=0; i<G[u].size(); i++)
    {
        int v=G[u][i];
        if(v==stop)continue;
        if(!pre[v]){
            child++;//未访问过的节点才能算是u的孩子
            int lowv=dfs(v,u,stop);
            lowu=min(lowu,lowv);
            if(lowv>=pre[u]){
                cut[u]++;
                iscut[u]=true;      //u点是割顶
                if(lowv>pre[u]) //割桥判定
                    Bridge.push_back(make_pair(u, v));
            }
        }
        else if(pre[v]<pre[u] && v!=fa){//v!=fa确保了(u,v)是从u到v的反向边
            lowu=min(lowu,pre[v]);
        }
    }
    if(fa<0 && child==1 )
        iscut[u]=false;//u若是根且孩子数<=1,那u就不是割顶
    return low[u]=lowu;
}

void solve(int x)
{

    memset(pre,0,sizeof(pre));
    memset(iscut,0,sizeof(iscut));
    memset(cut, 0, sizeof(cut));
    memset(low, 0, sizeof(low));
    Bridge.clear();
    dfs_clock=0;
    int k = 0;
    for (int i=1; i<=n; i++)
    {
        if(i==x) continue;
        if (!pre[i]) {
            k++;
            dfs(i,-1,x);//每次遍历一个连通块
        }
    } 
    for (int i=1; i<=n; i++)
    {
        if(i==x) continue;
        ans=max(ans,k+cut[i]);
    }
}
void work(){

    ans=-100;
    memset(pre,0,sizeof(pre));
    memset(iscut,0,sizeof(iscut));
    memset(cut, 0, sizeof(cut));
    memset(low, 0, sizeof(low));
    Bridge.clear();
    for(int i=0;i<=n;i++) G[i].clear();
    for(int i=0;i<m;i++){
        int u,v;
        scanf("%d%d",&u,&v);//下标1~n
        u++,v++;
        G[u].push_back(v);
        G[v].push_back(u);
    }
    for(int i=1;i<=n;i++)
        solve(i);
    printf("%d\n",ans);
}
int main()
{
    while(scanf("%d%d",&n,&m)!=EOF)
    {
        work();
    }
    return 0;
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值