1. 程式人生 > >洛谷 P3388 【模板】割點(割頂)

洛谷 P3388 【模板】割點(割頂)

tex def its next set clas pro != bit

P3388 【模板】割點(割頂)

題目背景

割點

題目描述

給出一個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

說明

n,m均為100000

tarjan 圖不一定聯通!!!、

沒錯我以前就是不會,不服bb別solo

#include<bits/stdc++.h>
using
namespace std; #define maxn 1000000 int n,m,x,y,tot,tim,dfn[maxn],low[maxn],head[maxn],ans; int cut_edge[maxn],cut_point[maxn]; bool vis[maxn]; struct Edge{ int next,to,from; }edge[maxn]; void add(int x,int y) { edge[tot].to=y; edge[tot].next=head[x]; head[x]=tot; tot++; }
void tarjan(int now,int pre) { int sum=0; bool if_=false; vis[now]=true; dfn[now]=low[now]=++tim; for(int i=head[now];i!=-1;i=edge[i].next) { int v=edge[i].to; if((i^1)!=pre) { if(!vis[v]) { sum++; tarjan(v,i);
if(low[v]>dfn[now]) cut_edge[i/2]=1; if(low[v]>=dfn[now]) if_=true; low[now]=min(low[now],low[v]); } else low[now]=min(low[now],dfn[v]); } } if(pre==-1) { if(sum>1) cut_point[now]=1; } else if(if_==1) cut_point[now]=1; return ; } int main() { scanf("%d%d",&n,&m); memset(head,-1,sizeof(head)); for(int i=1;i<=m;i++) scanf("%d%d",&x,&y),add(x,y),add(y,x); for(int i=1;i<=n;i++) if(!vis[i]) tarjan(i,-1); for(int i=1;i<=n;i++) if(cut_point[i]) ans++; printf("%d\n",ans); for(int i=1;i<=n;i++) if(cut_point[i]) printf("%d ",i); return 0; }

洛谷 P3388 【模板】割點(割頂)