1.题目
哥尼斯堡的“七桥问题”
分数 25
全屏浏览
切换布局
作者 DS课程组
单位 浙江大学
哥尼斯堡是位于普累格河上的一座城市,它包含两个岛屿及连接它们的七座桥,如下图所示。
可否走过这样的七座桥,而且每桥只走过一次?瑞士数学家欧拉(Leonhard Euler,1707—1783)最终解决了这个问题,并由此创立了拓扑学。
这个问题如今可以描述为判断欧拉回路是否存在的问题。欧拉回路是指不令笔离开纸面,可画过图中每条边仅一次,且可以回到起点的一条回路。现给定一个无向图,问是否存在欧拉回路?
输入格式:
输入第一行给出两个正整数,分别是节点数N (1≤N≤1000)和边数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
2.分析
由题目可知,我们要判断无向图中是否含有欧拉回路。
一个无向图含有欧拉回路有两个条件:
(1)无向图是连通的(没有单节点和独立的子图)
(2)无向图中的点的入度全是偶数
要判断图是否连通,可以用DFS(遍历所有结点)或并查集(所有结点的父亲相同)
我这里用的是并查集...(建议不会的先去学习一下)
3.代码
#include<iostream>
#include<vector>
using namespace std;
const int MAX=10000;
int N,M,rd[MAX],fa[MAX]; //fa[]并查集数组
//并查集find()函数
int find(int x){
if(fa[x]==x) return x;
else{
fa[x]=find(fa[x]);
return fa[x];
}
}
//并查集关联函数
void add(int a,int b){
int f_a=find(a);
int f_b=find(b);
fa[f_b]=f_a;
}
int main(){
cin>>N>>M;
int a,b;
for(int i=1;i<=N;i++){fa[i]=i;} //初始化并查集
for(int i=0;i<M;i++){
cin>>a>>b;
add(a,b); //关联结点
rd[a]++;
rd[b]++;
}
for(int i=1;i<=N;i++){
find(i); //缩短并查集路径(统一父结点)
if(rd[i]%2!=0){ //判断入度是否全为偶数
cout<<0<<endl;
return 0;
}
}
int f=fa[1];
for(int i=2;i<=N;i++){ //判断父结点是否一样(图是否连通)
if(fa[i]!=f){
cout<<0<<endl;
return 0;
}
}
cout<<1<<endl;
return 0;
}