POJ 2230 Watchcow(欧拉图)

Description
给出一张n个点的无向图,要求每条边都要走且只能走两次且方向相反,从点1开始走到点1,制定一条合理路径,保证有解
Input
第一行两个整数n和m表示点数和边数,之后m行每行两个整数a和b表示a点和b点之间有一条边(2<=n<=10000,1<=m<=50000)
Output
输出2*(m+1)个整数表示一条合理路径
Sample Input
4 5
1 2
1 4
2 3
2 4
3 4
Sample Output
1
2
3
4
2
1
4
3
2
4
1
Solution
无向图欧拉回路裸题,从1点出发,如果可以找到一条可以扩展的边(即没有被标记过)则标记这条边,然后从这条边的另一端继续搜索,点的回溯路径即为欧拉回路中点的序列,注意最后还要输出一个1
Code

#include<cstdio>
#include<iostream>
#include<algorithm>
#include<cstring>
using namespace std;
#define maxn 11111
struct node
{
    int to,next;
}edge[10*maxn];
int n,m,tot,head[maxn],vis[10*maxn],res,ans[10*maxn];
void init()
{
    tot=res=0;
    memset(head,-1,sizeof(head));
    memset(vis,0,sizeof(vis));
}
void add(int u,int v)
{
    edge[tot].to=v;
    edge[tot].next=head[u];
    head[u]=tot++;
}
void dfs(int u)
{
    for(int i=head[u];~i;i=edge[i].next)
    {
        if(!vis[i])
        {
            vis[i]=1;
            dfs(edge[i].to);
            ans[res++]=edge[i].to;
        }
    }
}
int main()
{
    while(~scanf("%d%d",&n,&m))
    {
        init();
        for(int i=0;i<m;i++)
        {
            int u,v;
            scanf("%d%d",&u,&v);
            add(u,v),add(v,u);
        }
        dfs(1);
        for(int i=0;i<res;i++)printf("%d\n",ans[i]);
        printf("1\n");
    }
    return 0;
}

你可能感兴趣的:(POJ 2230 Watchcow(欧拉图))