LSYOJ 电力 [DCC]

Author Avatar
空気浮遊 2018年08月22日
  • 在其它设备中阅读本文章

被坑了 n 个 WA

Problem

给一个无向图,删除某个点后问最大的连通块个数

Solution

处理出点双,割点共用的点双个数为 n 的话,会往原先连通块的个数 +n-1。

坑在于如果全部都是孤立点的话,连通块个数会 -1。

Code

// Code by ajcxsu
// Problem: electric

#include<bits/stdc++.h>
#define CLR(x) memset(x, 0, sizeof(x))
using namespace std;

const int N=1e6+10, M=4e6+10;
int h[N], to[M], nexp[M], p=1;
inline void ins(int a, int b) { nexp[p]=h[a], h[a]=p, to[p]=b, p++; }

vector<int> dcc[N];
int col[N], stk[N], dfn[N], low[N], t, idx, sidx, rt, cnt[N];
char cut[N];
void tarjan(int x) {
    dfn[x]=low[x]=++idx, stk[++t]=x;
    if(x==rt && !h[x]) { dcc[++sidx].push_back(x); return; }
    int cnt=0;
    for(int u=h[x];u;u=nexp[u])
        if(!dfn[to[u]]) {
            tarjan(to[u]), low[x]=min(low[x], low[to[u]]);
            if(low[to[u]]>=dfn[x]) {
                ++cnt;
                if(x!=rt || cnt>1) cut[x]=1;
                ++sidx;
                do {
                    dcc[sidx].push_back(stk[t]);
                } while(stk[t--]!=to[u]);
                dcc[sidx].push_back(x);
            }
        }
        else low[x]=min(low[x], dfn[to[u]]);
}

int main() {
    ios::sync_with_stdio(false);
    int n, m;
    while(cin>>n>>m, n) {
        CLR(h), p=1;
        int u, v;
        for(int i=0;i<m;i++) cin>>u>>v, ins(u, v), ins(v, u);
        sidx=idx=0, CLR(dfn), CLR(low), CLR(cut), CLR(cnt);
        for(int i=1;i<=n;i++) dcc[i].clear();
        int ini=0, ans=0;
        for(int i=0;i<n;i++)
            if(!dfn[i]) t=0, rt=i, tarjan(i), ini++;
        for(int i=1;i<=sidx;i++)
            for(int j=0, k=dcc[i].size(); j<k; j++) {
                v=dcc[i][j];
                if(cut[v]) cnt[v]++;
            }
        for(int i=0;i<n;i++)
            if(cut[i]) ans=max(ans, ini+cnt[i]-1);
        for(int i=1;i<=sidx;i++)
            ans=max(ans, ini-(dcc[i].size()==1));
        cout<<ans<<endl;
    }
    return 0;
}