P3469 : Tarjan 求割点以及割点的一些性质

题目

给你无向连通图,问你去掉一个点的所有关联边之后,满足 (x, y) 不连通有多少个有序点 ?

思路

tarjan() 求割点,如果当前点不是割点,去掉这个点之后是一个 n-1 连通图,但注意题目中不去掉这个点,那么当前点的答案就是 2 * (n - 1) ;

如果当前点是割点,假设割点切割出来 c 个连通块,思考如何 O(n) 算有序点对

首先是 O(n^2) 算,就是 c i c_i ci 去乘以所有非 i 连通块,手列一下式子很好化简,大概就是 ∑ i = 1 c C i ∗ ( n − C i ) \sum_{i=1}^cC_i*(n-C_i) ∑i=1cCi∗(n−Ci)

现在就是思考怎么求割出去的连通块大小 ?

tarjan() 过程中,如果 low[v]>=dfn[u],那么去掉 u 之后,v 会被分割出来一个 sz[v] 的连通块 ; 对于每一个 v 都是如此,去掉这些能被分割出来的连通块之后,剩下的图还是一个连通块

根据这个性质就好做了

cpp 复制代码
#include<bits/stdc++.h>
using namespace std;
#define int long long

int const N = 1e5 + 10;


int n, m;
vector<int> e[N];
int dfn[N], low[N], tot = 0;
bool is_Cut[N];
int ans[N], sz[N]; // 以 i 为根子树的大小

void tarjan(int u, bool isRoot){
    dfn[u] = low[u] = ++ tot;
    int sum = 0;
    sz[u] = 1;
    int childCount = 0;
    for(int v : e[u]){
        if(!dfn[v]){
            tarjan(v, false);
            sz[u] += sz[v];
            low[u] = min(low[u], low[v]);
            if(low[v] >= dfn[u]){
            	childCount ++;
            	sum += sz[v];
            	ans[u] += sz[v] * (n - sz[v]);
            }
        }
        else{
            low[u] = min(low[u], dfn[v]);
        }
    }
    if(childCount > isRoot){
        is_Cut[u] = true;
        ans[u] += n - 1;
        ans[u] += (n - sum - 1) * (sum + 1);
    }
    else{
    	ans[u] = 2 * (n - 1);
    }
}
void tarjan(){
    for(int i = 1; i <= n; i ++){
        if(!dfn[i]){
            tarjan(i, true);
        }
    }
}

void solve(){
	cin >> n >> m;
	for(int i = 1; i <= m; i ++){
		int u, v;
		cin >> u >> v;
		e[u].push_back(v);
		e[v].push_back(u);
	}
	tarjan();
	for(int i = 1; i <= n; i ++){
		cout << ans[i] << '\n';
    }
}

signed main(){
	ios::sync_with_stdio(false);
	cin.tie(0), cout.tie(0); 
	solve();
	return 0;
}
相关推荐
无敌最俊朗@5 小时前
力扣hot100-206反转链表
算法·leetcode·链表
Kuo-Teng5 小时前
LeetCode 279: Perfect Squares
java·数据结构·算法·leetcode·职场和发展
王哈哈^_^5 小时前
YOLO11实例分割训练任务——从构建数据集到训练的完整教程
人工智能·深度学习·算法·yolo·目标检测·机器学习·计算机视觉
檐下翻书1736 小时前
从入门到精通:流程图制作学习路径规划
论文阅读·人工智能·学习·算法·流程图·论文笔记
CoderYanger6 小时前
B.双指针——3194. 最小元素和最大元素的最小平均值
java·开发语言·数据结构·算法·leetcode·职场和发展·1024程序员节
小曹要微笑7 小时前
STM32各系列时钟树详解
c语言·stm32·单片机·嵌入式硬件·算法
前进的李工8 小时前
LeetCode hot100:094 二叉树的中序遍历:从递归到迭代的完整指南
python·算法·leetcode·链表·二叉树
麦麦大数据9 小时前
F049 知识图谱双算法推荐在线学习系统vue+flask+neo4j之BS架构开题论文全源码
学习·算法·知识图谱·推荐算法·开题报告·学习系统·计算机毕业设计展示
兩尛10 小时前
215. 数组中的第K个最大元素
数据结构·算法·排序算法
9523610 小时前
数据结构-堆
java·数据结构·学习·算法