洛谷 P3388 模板割点(割顶)
Posted 一蓑烟雨任生平
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了洛谷 P3388 模板割点(割顶)相关的知识,希望对你有一定的参考价值。
题目背景
割点
题目描述
给出一个n个点,m条边的无向图,求图的割点。
输入输出格式
输入格式:
第一行输入n,m
下面m行每行输入x,y表示x到y有一条边
输出格式:
第一行输出割点个数
第二行按照节点编号从小到大输出节点,用空格隔开
输入输出样例
说明
n,m均为100000
tarjan 图不一定联通!!!
#include<cstdio> #include<cstring> #include<iostream> #include<algorithm> #define MAXN 100010 using namespace std; int n,m,ans; int tot=1,tim; int cutdian[MAXN],cutbian[MAXN]; int to[MAXN*2],net[MAXN*2],head[MAXN]; int low[MAXN],dfn[MAXN],vis[MAXN]; void add(int u,int v){ to[++tot]=v;net[tot]=head[u];head[u]=tot; to[++tot]=u;net[tot]=head[v];head[v]=tot; } void tarjin(int now,int pre){ low[now]=dfn[now]=++tim; vis[now]=1; int sum=0; bool boo=0; for(int i=head[now];i!=-1;i=net[i]) if((i^1)!=pre) if(!vis[to[i]]){ sum++; tarjin(to[i],i); if(low[to[i]]>dfn[now]) cutbian[i/2]=1; if(low[to[i]]>=dfn[now]) boo=1; low[now]=min(low[now],low[to[i]]); } else low[now]=min(low[now],dfn[to[i]]); if(pre==-1){ if(sum>1) cutdian[now]=1; } else if(boo==1) cutdian[now]=1; } int main(){ memset(head,-1,sizeof(head)); scanf("%d%d",&n,&m); for(int i=1;i<=m;i++){ int u,v; scanf("%d%d",&u,&v); add(u,v); } for(int i=1;i<=n;i++) if(!vis[i]) tarjin(i,-1); for(int i=1;i<=n;i++) if(cutdian[i]) ans++; cout<<ans<<endl; for(int i=1;i<=n;i++) if(cutdian[i]) cout<<i<<" "; }
以上是关于洛谷 P3388 模板割点(割顶)的主要内容,如果未能解决你的问题,请参考以下文章