首页 > 其他 > 详细

BZOJ 1123 tarjan

时间:2018-09-18 21:58:57      阅读:178      评论:0      收藏:0      [点我收藏+]

题目链接

题意:一张无向图,把第$i$个点关联的所有边去掉,求无向图中有多少个点对不连通。

题解:

如果割的不是割点,那么总答案是$2\times (n-1)$.

如果是割点,要分别考虑每个子树的贡献. 

技术分享图片
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
using namespace std;
typedef long long ll;
const int N = 1e5 + 10, M = 5e5 + 10;
int head[N], ver[M * 2], Next[M * 2];
int dfn[N], low[N], Size[N];
ll ans[N];
bool cut[N];
int n, m, tot, num; ///时间戳号

void add(int u, int v)
{
    ver[++tot] = v, Next[tot] = head[u], head[u] = tot;
}
void tarjan(int u)
{
    dfn[u] = low[u] = ++num; Size[u] = 1;
    int flag = 0, sum = 0;
    for(int i = head[u]; i; i = Next[i])
    {
        int v = ver[i];
        if(!dfn[v])
        {
            tarjan(v);
            Size[u] += Size[v];
            low[u] = min(low[u], low[v]);
            if(low[v] >= dfn[u])
            {
                flag++;
                ans[u] += (ll)Size[v] * (n - Size[v]);
                sum += Size[v];
                if(u != 1 || flag > 1) cut[u] = true;
            }
        }
        else low[u] = min(low[u], dfn[v]);
    }
    if(cut[u]) ans[u] += (ll)(n - sum - 1) * (sum + 1) + (n - 1);
    else ans[u] = 2 * (n - 1);
}
int main()
{
    scanf("%d %d", &n, &m); tot = 1;
    for(int i = 1; i <= m; i++)
    {
        int u, v; scanf("%d %d", &u, &v);
        if(u == v) continue;
        add(u, v), add(v, u);
    }
    tarjan(1);
    for(int i = 1; i <= n; i++) printf("%lld\n", ans[i]);
    return 0;
}
Code

 

BZOJ 1123 tarjan

原文:https://www.cnblogs.com/wangwangyu/p/9671083.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!