7-23 哥尼斯堡的“七桥问题”(25 point(s))

7-23 哥尼斯堡的“七桥问题”(25 point(s))

哥尼斯堡是位于普累格河上的一座城市,它包含两个岛屿及连接它们的七座桥,如下图所示。

7-23 哥尼斯堡的“七桥问题”(25 point(s))_第1张图片

可否走过这样的七座桥,而且每桥只走过一次?瑞士数学家欧拉(Leonhard Euler,1707—1783)最终解决了这个问题,并由此创立了拓扑学。

这个问题如今可以描述为判断欧拉回路是否存在的问题。欧拉回路是指不令笔离开纸面,可画过图中每条边仅一次,且可以回到起点的一条回路。现给定一个无向图,问是否存在欧拉回路?

输入格式:

输入第一行给出两个正整数,分别是节点数N (1N1000)和边数M;随后的M行对应M条边,每行给出一对正整数,分别是该条边直接连通的两个节点的编号(节点从1到N编号)。

输出格式:

若欧拉回路存在则输出1,否则输出0。

输入样例1:

6 10
1 2
2 3
3 1
4 5
5 6
6 4
1 4
1 6
3 4
3 6

输出样例1:

1

输入样例2:

5 8
1 2
1 3
2 3
2 4
2 5
5 3
5 4
3 4

输出样例2:

0


判断是否是欧拉回路的题

通过图中所有边一次且仅一次行遍所有顶点是回路,称为欧拉回路。。

具有欧拉回路的图称为欧拉图。。

解这个问题,非常的简单,就需要一个定理就好了。

无向图是欧拉图当且仅当图是连通图且没有奇度顶点。。。

那么,我们就仅仅需要判断这个图的连通性和顶点的度数就好啦。。。

Code:

//无向图判断是否是欧拉回路,首先是连通图其次不存在奇度顶点
//因此只需要判断两个,用并查集看是否连通,然后遍历每个点看度数是否全为偶数
#include 
#include 
int pre[1002];//祖先数组 
int degree[1002];//记录每个点的度数
int n,m;
void init(){
	int i;
	for(i = 0; i < 1002; i++){
		pre[i] = i;
		degree[i] = 0;
	}
}//初始化 
int Find(int x){//并查集查找 
	if(x==pre[x])return x;
	else return pre[x] = Find(pre[x]);
} 
void Union(int x,int y){
	int fx = Find(x),fy = Find(y);
	if(fx!=fy){
		pre[fy] = fx;
		return ;
	}
	return;
}
int solve(){
	int i,t = 0;//t用来统计父亲个数,父亲只要一个说明是连通图 
	for(i = 1; i <= n; i++){
		if(pre[i]==i)t++;
	}
	if(t!=1)return 0;//不连通返回0
	for(i = 1; i <= n; i++){
		if(degree[i]&1)return 0;//存在奇度顶点返回0 
	} 
	return 1;//最后都满足返回1 
}
int main(){
	int i,x,y;
	init();//先初始化两个数组 
	scanf("%d%d",&n,&m);
	for(i = 1; i <= m; i++){
		scanf("%d%d",&x,&y);
		Union(x,y);//并查集连接 
		degree[x]++;//记录每个点的度数 
		degree[y]++; 
	} 
	if(solve())printf("1\n");
	else printf("0\n");
	return 0;
} 


你可能感兴趣的:(数据结构,欧拉回路)