HDU 1878(1Y) (判断欧拉回路是否存在 奇点个数为0 + 一个联通分量 *【模板】)

欧拉回路

Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)
Total Submission(s): 9797    Accepted Submission(s): 3554

Problem Description

欧拉回路是指不令笔离开纸面,可画过图中每条边仅一次,且可以回到起点的一条回路。现给定一个图,问是否存在欧拉回路?

Input

测试输入包含若干测试用例。每个测试用例的第1行给出两个正整数,分别是节点数N ( 1 < N < 1000 )和边数M;随后的M行对应M条边,每行给出一对正整数,分别是该条边直接连通的两个节点的编号(节点从1到N编号)。当N为0时输入结
束。

Output

每个测试用例的输出占一行,若欧拉回路存在则输出1,否则输出0。

Sample Input

3 3

1 2

1 3

2 3

3 2

1 2

2 3

0

Sample Output

1

0

算法分析:此题考察欧拉回路是否存在。要知道欧拉回路和欧拉道路是不一样的。欧拉回路存在则一定存在欧拉道路,但是欧拉道路存在则不一定存在欧拉回路!

因为欧拉道路不一定是一个回路,也就是说,只要能把所有的节点连接起来就是一条欧拉道路了,但不一定能顺利回到起点。

关于欧拉路径问题的理论总结:

1)能否从一个无向图的一个节点出发走出一条道路,每条边恰好经过一次,这样的路线成为欧拉回路。“就是一个一笔画”

2)欧拉道路存在定理:如果一个无向图是连通的,且最多只有2个奇点,则一定存在。  (奇点:节点度数为奇数的点)

如果有2个奇点,则必须从其中一个出发,另一个奇点终止。

如果奇点不存在,则可以从任意节点出发,最终定会回到该点。

3)有向图下的理论:

大前提:  在忽略边的方向后,图必须是连通的,否则不可能存在欧拉道路。

有向图最多只能有两个节点的入度不等于出度,而且必须是出度比入度大1的节点做起点,入度比出度大1的节点做终点。

此题的代码:

#include <stdio.h>
#include <string.h>
#include <iostream>
#include <string>
#include <algorithm>
#define N 1000+2

using namespace std;
bool map[N][N];
int odd[N];
bool vis[N];
int n;

void dfs(int dd) //测试图的连通性
{
    for(int i=1; i<=n; i++)
    {
        if(!vis[i] && map[dd][i]==true )
        {
            vis[i]=true;
            dfs(i);
        }
    }
}

int main()
{
    int m;
    int i, j;
    int dd; //计数奇点的数目
    int u,v;

    while(scanf("%d", &n)!=EOF)
    {
        if(n==0)
            break;
        scanf("%d", &m);
        for(i=0; i<=n; i++)
        {
            for(j=0; j<=n; j++)
            {
                map[i][j]=false;
            }
        }
        memset(odd, 0, sizeof(odd));

        for(i=0; i<m; i++)
        {
            scanf("%d %d", &u, &v);
            map[u][v]=true;
            map[v][u]=true;
            odd[u]++;
            odd[v]++;
        }
        for(i=1; i<=n; i++)
            vis[i]=false;

        int ans=0; //计算连通分量的个数
        for(i=1; i<=n; i++)
        {
            if(!vis[i])
            {
                ans++;
                vis[i]=true;
                dfs(i);
            }
        }
        if(ans==1) //连通分量的个数为1,说明图是连通的
        {
            dd=0;
            for(i=1; i<=n; i++ )
            {
                if(odd[i]%2==1) //统计奇点的个数
                    dd++;
            }
            if( dd==0 ) //如果奇点的个数为0,则说明存在欧拉回路
                printf("1\n");
            else
                printf("0\n");
        }
        else
            printf("0\n");
    }
    return 0;
}
时间: 2025-01-14 16:34:45

HDU 1878(1Y) (判断欧拉回路是否存在 奇点个数为0 + 一个联通分量 *【模板】)的相关文章

爆零后的感受外加一道强联通分量HDU 4635的题解

今天又爆零了,又是又,怎么又是又,爆零爆多了,又也就经常挂嘴边了,看到这句话,你一定很想说一句"",弱菜被骂傻,也很正常啦. 如果你不开心,可以考虑往下看. 翻到E(HDU 4635 Strongly connected)题,这么短的题目,肯定要先看啦.然后D(LightOJ 1229),然后C(ZOJ 2243),然后F(HDU 4711),然后B(CodeForces 385D),然后看A(HDU 3889)好吧,我承认,A题看了一眼就不看了,B题一看是线段什么有点几何的味道就果断

HDU 1878 欧拉回路 (并查集+欧拉回路)

题目地址:HDU 1878 这个题要注意欧拉回路与欧拉通路的区别.在都保证连通性的前提下,欧拉回路要求每个点的度数都是偶数,而欧拉通路允许两个点的度数是奇数.所以这题用并查集判断连通性后判断下度数就可以了. 代码如下: #include <iostream> #include <string.h> #include <math.h> #include <queue> #include <algorithm> #include <stdlib

hdu 1878 欧拉回路 (欧拉回路)

//1.联通图 2.顶点度数都为偶数 ,则存在欧拉回路. # include <stdio.h> # include <algorithm> # include <string.h> using namespace std; int father[1010]; int vis[1010][1010],du[1010]; int find(int x) { if(father[x]==x) return x; return father[x]=find(father[x]

HDU 1878.欧拉回路【欧拉路及欧拉回路初接触】【8月2】

欧拉回路 Problem Description 欧拉回路是指不令笔离开纸面,可画过图中每条边仅一次,且可以回到起点的一条回路.现给定一个图,问是否存在欧拉回路? Input 测试输入包含若干测试用例.每个测试用例的第1行给出两个正整数,分别是节点数N ( 1 < N < 1000 )和边数M:随后的M行对应M条边,每行给出一对正整数,分别是该条边直接连通的两个节点的编号(节点从1到N编号).当N为0时输入结 束. Output 每个测试用例的输出占一行,若欧拉回路存在则输出1,否则输出0.

[2016-01-21][HDU][1878]

/************************************************************* 时间:2016-01-20  15:00:34  星期三 题目编号:G HDU 1878 题目大意:给定一个图,判断是否是欧拉图 方法:         1.度数是偶数        2.连通量为1 判断连通量 可以 用 并查集 也可以dfs 解题过程遇到问题: *******************************************************

poj1300判断欧拉回路

对于连通图 无向图:1.无奇点,可以从任意一点出发回到原点. 2.存在奇点,且只有两个,从一奇点出发,另一奇点终止. 有向图:1.所有点入度等于出度. 2.只有两个点入度不等于出度,且其中一个点入度比出度大一另一个点的出度比入度大一. #include <cstdio> #include <cstdlib> #include <cstring> #include <algorithm> #include <cmath> #include <

编写一个宏,实现判断数组a元素的个数

#include <iostream> using namespace std; #define TestArrayLengthA(A) sizeof(A)/sizeof(*A) #define TestArrayLengthB(B) sizeof(B)/sizeof(B[0]) //这样测出的是数组可以放多少个元素,比如Array[100],他返回的是100, //不论你初始化还是没有初始化 int TestArrayLength(T *a) { int count = 0; T *p =

hdu 5919 主席树(区间不同数的个数 + 区间第k大)

Sequence II Time Limit: 9000/4500 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others)Total Submission(s): 849    Accepted Submission(s): 204 Problem Description Mr. Frog has an integer sequence of length n, which can be denoted as a1,a2,?

hdu 4353 统计点在三角形内的个数

Finding Mine Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submission(s): 1120    Accepted Submission(s): 298 Problem Description In bitland, there is an area with M gold mines. As a businessman, Bob wants t