MYOJ_7789:(洛谷P3388)【模板】割点(割顶)(tarjan算法)

题目描述

给出一个 n 个点,m 条边的无向图,求图的割点。

输入

第一行输入两个正整数 n,m。

下面 m 行每行输入两个正整数 x,y 表示 x 到 y 有一条边。

输出

第一行输出割点个数。

第二行按照节点编号从小到大输出节点,用空格隔开。

样例输入输出

输入 #1

复制代码
6 7
1 2
1 3
1 4
2 5
3 5
4 5
5 6

输出 #1

复制代码
1
5
思路

对于割点问题,我们仍然使用tarjan算法完成。

所谓割点,就是在无向连通图中,删除该节点以及所有与它相连的边,会使图变得不连通(即连通分量数增加)的点。

在无向图中,DFS生成树包含树边和回边。

这里存在两种情况:

1.如果根节点u是割点,则u至少有两个子节点。 (如果只有一个,那去掉之后并不影响图的连通性)

2.如果非根节点u是割点,则存在一个u的孩子v,使得dfn[u]****≤low[v]。

(即v及其子树中所有节点不通过u无法到达u的祖先 )

所以我们只需要对tarjan稍作修改即可。

这一次tarjan需要用到以下变量:

cpp 复制代码
int n,m,u,v,vn;
int dfn[N],low[N],ts,root;
set<int>st;
vector<int>edge[N];
/*
dfn[N]:时间戳
low[N]:追溯值
ts:时间戳计数
root:当前连通分量的DFS起点
st:存储割点
*/

tarjan与强连通分量的版本大同小异。

cpp 复制代码
void tarjan(int u)
{
	int child=0;//child用来保存孩子数量(根节点)
	dfn[u]=low[u]=++ts;
	for(int v:edge[u])
	{
		if(dfn[v]==0)//树边
		{
			tarjan(v);
			low[u]=min(low[u],low[v]);
			if(u==root&&++child>=2||u!=root&&dfn[u]<=low[v])//是割点,详见上文
			{
				st.insert(u);
			}
		}
		else//回边
		{
			low[u]=min(low[u],dfn[v]);//再次强调!!!
		}
	}
}

现在有了前面的分析,可以分步写代码了。

STEP 1:根据定义写tarjan,这里不再赘述。

STEP 2:输入,建好无向图,枚举每个节点,对每一个连通分量都要进行tarjan。

STEP 3:在tarjan中的所有割点信息已经在set中自动排序去重,直接输出set的长度和元素即可。

代码
cpp 复制代码
#include<bits/stdc++.h>
#define N 20005
#define Letian 14
using namespace std;
int n,m,u,v,vn;
int dfn[N],low[N],ts,root;
set<int>st;
vector<int>edge[N];
void tarjan(int u)
{
	int child=0;
	dfn[u]=low[u]=++ts;
	for(int v:edge[u])
	{
		if(dfn[v]==0)
		{
			tarjan(v);
			low[u]=min(low[u],low[v]);
			if(u==root&&++child>=2||u!=root&&dfn[u]<=low[v])
			{
				st.insert(u);
			}
		}
		else
		{
			low[u]=min(low[u],dfn[v]);
		}
	}
}
int main()
{
    ios::sync_with_stdio(false);
    cin.tie(0);
    cout.tie(0);
    cin>>n>>m;
    for(int i=1;i<=m;i++)
    {
    	cin>>u>>v;
    	edge[u].push_back(v);
    	edge[v].push_back(u);
	}
	for(int i=1;i<=n;i++)
	{
		if(dfn[i]==0)
		{
			tarjan(root=i);
		}
	}
	cout<<st.size()<<'\n';
	for(int v:st)
	{
		cout<<v<<' ';
	}
    return 0;
}
运行结果

感谢阅读,我们下期再会。

如果您喜欢本文,请您点赞、收藏加关注,以防你找不到回来的路,谢谢!

相关推荐
DuHz6 小时前
论文精读:大语言模型 (Large Language Models, LLM) —— 一项调查
论文阅读·人工智能·深度学习·算法·机器学习·计算机视觉·语言模型
加农炮手Jinx6 小时前
LeetCode 72. Edit Distance 题解
算法·leetcode·力扣
借雨醉东风6 小时前
程序分享--常见算法/编程面试题:旋转矩阵
c++·线性代数·算法·面试·职场和发展·矩阵
_深海凉_6 小时前
LeetCode热题100-打家劫舍
算法·leetcode·职场和发展
jghhh017 小时前
使用 MATLAB 实现支持向量回归 (SVR) 预测未来数据
算法·matlab
云泽8087 小时前
笔试算法 - 双指针篇(二):四大经典求和题型 + 有效三角形计数问题
c++·算法
十五年专注C++开发8 小时前
WaitingSpinnerWidget: 一个高度可配置的自定义Qt等待加载动画组件
开发语言·c++·qt·waitingspinner
qeen878 小时前
【数据结构】树的基本概念及存储
c语言·数据结构·c++·学习·
刀法如飞8 小时前
【合并已排序数组的三种实现策略,哪一种更可取?】
算法·程序员
王老师青少年编程8 小时前
csp信奥赛C++高频考点专项训练之贪心算法 --【区间贪心】:种树
c++·算法·贪心·csp·信奥赛·区间贪心·种树