模拟考 T3 Tree

Description

Fanvree很聪明,解决难题时他总会把问题简单化。例如,他就整天喜欢把图转化为树。但是他不会缩环,那他怎么转化呢? 这是一个有n个点m条双向边的图,Fanvree会选定一个节点,然后删掉这个节点和这个点连出去的边,如果变成了一棵树,那么这个节点便是可行的,什么是树呢?树也即无简单环的无向连通图。告诉Fanvree可能的节点是什么。

Input

第一行两个正整数n和m,表示有n个点m条边,保证n≥2。接下来m行,每行两个整数v,u,表示v和u之间有一条无向边1≤v,u≤n,保证没有重边和自环。

Output

第一行一个正整数ns,表示这个图中有ns个结点可选。接下来一行,共ns个整数,每个整数表示一个可选结点的编号。请按编号从小到大的顺序输出。数据保证图中至少存在一个可选的结点。

Hint

对于40%的数据:n,m<=1000; 另外存在10%的数据:m=n-1; 另外存在20%的数据:m=n; 对于100%的数据:n,m<=100000。

Solution

预处理出割点,统计每个点的度,如果m-这个点的度==n-2并且这个点不是割点,那么它就可以删。
#include
#include
#include
#include
#define maxn 200005
using namespace std;
struct Edge{
    int u;
    int v;
    int next;
    int id;
}edge[maxn];
int first[maxn],last[maxn],low[maxn],dfn[maxn],du[maxn],ans[maxn];
int node,n,m,x,y,dfn_TimeClock,cnt;
bool vis[maxn],cut[maxn];
void addedge(int u,int v,int id){
    edge[++node]=(Edge){u,v,0,id};
    if(first[u]==0)first[u]=node;
    else edge[last[u]].next=node;
    last[u]=node;
}
void init(){
    scanf("%d%d",&n,&m);
    for(int i=1;i<=m;i++){
        scanf("%d%d",&x,&y);
        du[x]++;
        du[y]++;
        addedge(x,y,i);
        addedge(y,x,i);
    }
}
void Tarjan_h(int i,int fd){
    vis[i]=true;
    dfn[i]=low[i]=++dfn_TimeClock;
    int xxx=0;
    for(int p=first[i];p;p=edge[p].next){
        int j=edge[p].v,id=edge[p].id;
        if(vis[j]){
            if(dfn[j]=dfn[i])cut[i]=1;
    }
    if(xxx==1&&fd==0)cut[i]=0;
}
void workk(){
    Tarjan_h(1,0);
    for(int i=1;i<=n;i++){
        if(m-du[i]==n-2&&!cut[i]){
            ans[++cnt]=i;
        }
    }
}
int main(){
    init();
    workk();
    printf("%d\n",cnt);
    for(int i=1;i<=cnt;i++){
        printf("%d ",ans[i]);
    }
    printf("\n");
    return 0;
}

转载于:https://www.cnblogs.com/virtual-north-Illya/p/10045344.html

你可能感兴趣的:(模拟考 T3 Tree)