【PTA数据结构 | C语言版】哥尼斯堡的“七桥问题”

本专栏持续输出数据结构题目集,欢迎订阅。

文章目录

题目

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

可否走过这样的七座桥,而且每桥只走过一次?瑞士数学家欧拉(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

代码

cpp 复制代码
#include <stdio.h>
#include <stdlib.h>

#define MAX_N 1000

// 邻接表节点结构
typedef struct Node {
    int vertex;
    struct Node* next;
} Node;

// 并查集结构
typedef struct {
    int parent[MAX_N + 1];
    int rank[MAX_N + 1];
} UnionFind;

// 初始化并查集
void initUnionFind(UnionFind* uf, int n) {
    for (int i = 1; i <= n; i++) {
        uf->parent[i] = i;
        uf->rank[i] = 1;
    }
}

// 查找根节点并路径压缩
int find(UnionFind* uf, int x) {
    if (uf->parent[x] != x) {
        uf->parent[x] = find(uf, uf->parent[x]);
    }
    return uf->parent[x];
}

// 合并两个集合
void unionSets(UnionFind* uf, int x, int y) {
    int xRoot = find(uf, x);
    int yRoot = find(uf, y);
    
    if (xRoot == yRoot) return;
    
    if (uf->rank[xRoot] < uf->rank[yRoot]) {
        uf->parent[xRoot] = yRoot;
    } else if (uf->rank[xRoot] > uf->rank[yRoot]) {
        uf->parent[yRoot] = xRoot;
    } else {
        uf->parent[yRoot] = xRoot;
        uf->rank[xRoot]++;
    }
}

int main() {
    int n, m;
    scanf("%d %d", &n, &m);
    
    // 记录每个顶点的度数
    int degree[MAX_N + 1] = {0};
    
    // 初始化并查集
    UnionFind uf;
    initUnionFind(&uf, n);
    
    // 处理每条边
    for (int i = 0; i < m; i++) {
        int u, v;
        scanf("%d %d", &u, &v);
        
        // 更新度数
        degree[u]++;
        degree[v]++;
        
        // 合并两个顶点所在的集合
        unionSets(&uf, u, v);
    }
    
    // 检查所有边是否在同一个连通分量中
    int root = -1;
    int hasEdge = 0;
    
    for (int i = 1; i <= n; i++) {
        if (degree[i] > 0) {
            hasEdge = 1;
            if (root == -1) {
                root = find(&uf, i);
            } else if (find(&uf, i) != root) {
                // 存在多个连通分量
                printf("0\n");
                return 0;
            }
        }
    }
    
    // 如果没有边,特殊处理
    if (!hasEdge) {
        printf("0\n");
        return 0;
    }
    
    // 检查所有顶点的度数是否为偶数
    for (int i = 1; i <= n; i++) {
        if (degree[i] % 2 != 0) {
            printf("0\n");
            return 0;
        }
    }
    
    // 所有条件满足,存在欧拉回路
    printf("1\n");
    
    return 0;
}    
相关推荐
汀、人工智能6 小时前
[特殊字符] 第77课:最长递增子序列
数据结构·算法·数据库架构·图论·bfs·最长递增子序列
网域小星球6 小时前
C语言从0入门(十)|二维数组详解与矩阵实战
c语言·算法·矩阵·二维数组·数组遍历
网域小星球6 小时前
C 语言从 0 入门(十五)|综合小项目:菜单交互与简易功能实现
c语言·开发语言·交互
澈2077 小时前
堆排序:高效构建大顶堆实战
数据结构·算法·排序算法
网域小星球7 小时前
C 语言从 0 入门(十六)|动态内存管理:malloc /free/calloc /realloc 精讲
c语言·开发语言·free·malloc·动态内存
chh5637 小时前
C++--内存管理
java·c语言·c++·windows·学习·面试
我真不是小鱼7 小时前
cpp刷题打卡记录27——无重复字符的最长子串 & 找到字符串中所有字母的异位词
数据结构·c++·算法·leetcode
XuecWu37 小时前
原生多模态颠覆Scaling Law?解读语言“参数需求型”与视觉“数据需求型”核心差异
人工智能·深度学习·算法·计算机视觉·语言模型
We་ct7 小时前
LeetCode 69. x 的平方根:两种解法详解
前端·javascript·算法·leetcode·typescript·平方
一直不明飞行7 小时前
C++:string,写法s.find(‘@‘) != s.end()是否有问题
开发语言·c++·算法