哥尼斯堡的“七桥问题”(简单算法)

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;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值