|
| 1 | + |
| 2 | +# 107. 寻找存在的路径 |
| 3 | + |
| 4 | +[卡码网题目链接(ACM模式)](https://kamacoder.com/problempage.php?pid=1179) |
| 5 | + |
| 6 | +题目描述 |
| 7 | + |
| 8 | +给定一个包含 n 个节点的无向图中,节点编号从 1 到 n (含 1 和 n )。 |
| 9 | + |
| 10 | +你的任务是判断是否有一条从节点 source 出发到节点 destination 的路径存在。 |
| 11 | + |
| 12 | +输入描述 |
| 13 | + |
| 14 | +第一行包含两个正整数 N 和 M,N 代表节点的个数,M 代表边的个数。 |
| 15 | + |
| 16 | +后续 M 行,每行两个正整数 s 和 t,代表从节点 s 与节点 t 之间有一条边。 |
| 17 | + |
| 18 | +最后一行包含两个正整数,代表起始节点 source 和目标节点 destination。 |
| 19 | + |
| 20 | +输出描述 |
| 21 | + |
| 22 | +输出一个整数,代表是否存在从节点 source 到节点 destination 的路径。如果存在,输出 1;否则,输出 0。 |
| 23 | + |
| 24 | +输入示例 |
| 25 | + |
| 26 | +``` |
| 27 | +5 4 |
| 28 | +1 2 |
| 29 | +1 3 |
| 30 | +2 4 |
| 31 | +3 4 |
| 32 | +1 4 |
| 33 | +``` |
| 34 | + |
| 35 | +输出示例 |
| 36 | + |
| 37 | +1 |
| 38 | + |
| 39 | +提示信息 |
| 40 | + |
| 41 | + |
| 42 | + |
| 43 | +数据范围: |
| 44 | + |
| 45 | +1 <= M, N <= 100。 |
| 46 | + |
| 47 | +## 思路 |
| 48 | + |
| 49 | +本题是并查集基础题目。 如果还不了解并查集,可以看这里:[并查集理论基础](https://programmercarl.com/kamacoder/图论并查集理论基础.html) |
| 50 | + |
| 51 | +并查集可以解决什么问题呢? |
| 52 | + |
| 53 | +主要就是集合问题,**两个节点在不在一个集合,也可以将两个节点添加到一个集合中**。 |
| 54 | + |
| 55 | +这里整理出我的并查集模板如下: |
| 56 | + |
| 57 | +```CPP |
| 58 | +int n = 1005; // n根据题目中节点数量而定,一般比节点数量大一点就好 |
| 59 | +vector<int> father = vector<int> (n, 0); // C++里的一种数组结构 |
| 60 | + |
| 61 | +// 并查集初始化 |
| 62 | +void init() { |
| 63 | + for (int i = 0; i < n; ++i) { |
| 64 | + father[i] = i; |
| 65 | + } |
| 66 | +} |
| 67 | +// 并查集里寻根的过程 |
| 68 | +int find(int u) { |
| 69 | + return u == father[u] ? u : father[u] = find(father[u]); // 路径压缩 |
| 70 | +} |
| 71 | + |
| 72 | +// 判断 u 和 v是否找到同一个根 |
| 73 | +bool isSame(int u, int v) { |
| 74 | + u = find(u); |
| 75 | + v = find(v); |
| 76 | + return u == v; |
| 77 | +} |
| 78 | + |
| 79 | +// 将v->u 这条边加入并查集 |
| 80 | +void join(int u, int v) { |
| 81 | + u = find(u); // 寻找u的根 |
| 82 | + v = find(v); // 寻找v的根 |
| 83 | + if (u == v) return ; // 如果发现根相同,则说明在一个集合,不用两个节点相连直接返回 |
| 84 | + father[v] = u; |
| 85 | +} |
| 86 | +``` |
| 87 | +
|
| 88 | +以上模板中,只要修改 n 大小就可以。 |
| 89 | +
|
| 90 | +并查集主要有三个功能: |
| 91 | +
|
| 92 | +1. 寻找根节点,函数:find(int u),也就是判断这个节点的祖先节点是哪个 |
| 93 | +2. 将两个节点接入到同一个集合,函数:join(int u, int v),将两个节点连在同一个根节点上 |
| 94 | +3. 判断两个节点是否在同一个集合,函数:isSame(int u, int v),就是判断两个节点是不是同一个根节点 |
| 95 | +
|
| 96 | +简单介绍并查集之后,我们再来看一下这道题目。 |
| 97 | +
|
| 98 | +为什么说这道题目是并查集基础题目,题目中各个点是双向图链接,那么判断 一个顶点到另一个顶点有没有有效路径其实就是看这两个顶点是否在同一个集合里。 |
| 99 | +
|
| 100 | +如何算是同一个集合呢,有边连在一起,就算是一个集合。 |
| 101 | +
|
| 102 | +此时我们就可以直接套用并查集模板。 |
| 103 | +
|
| 104 | +使用 join(int u, int v)将每条边加入到并查集。 |
| 105 | +
|
| 106 | +最后 isSame(int u, int v) 判断是否是同一个根 就可以了。 |
| 107 | +
|
| 108 | +C++代码如下: |
| 109 | +
|
| 110 | +```CPP |
| 111 | +#include <iostream> |
| 112 | +#include <vector> |
| 113 | +using namespace std; |
| 114 | +
|
| 115 | +int n; // 节点数量 |
| 116 | +vector<int> father = vector<int> (101, 0); // 按照节点大小定义数组大小 |
| 117 | +
|
| 118 | +// 并查集初始化 |
| 119 | +void init() { |
| 120 | + for (int i = 1; i <= n; i++) father[i] = i; |
| 121 | +} |
| 122 | +// 并查集里寻根的过程 |
| 123 | +int find(int u) { |
| 124 | + return u == father[u] ? u : father[u] = find(father[u]); |
| 125 | +} |
| 126 | +
|
| 127 | +// 判断 u 和 v是否找到同一个根 |
| 128 | +bool isSame(int u, int v) { |
| 129 | + u = find(u); |
| 130 | + v = find(v); |
| 131 | + return u == v; |
| 132 | +} |
| 133 | +
|
| 134 | +// 将v->u 这条边加入并查集 |
| 135 | +void join(int u, int v) { |
| 136 | + u = find(u); // 寻找u的根 |
| 137 | + v = find(v); // 寻找v的根 |
| 138 | + if (u == v) return ; // 如果发现根相同,则说明在一个集合,不用两个节点相连直接返回 |
| 139 | + father[v] = u; |
| 140 | +} |
| 141 | +
|
| 142 | +int main() { |
| 143 | + int m, s, t, source, destination; |
| 144 | + cin >> n >> m; |
| 145 | + init(); |
| 146 | + while (m--) { |
| 147 | + cin >> s >> t; |
| 148 | + join(s, t); |
| 149 | + } |
| 150 | + cin >> source >> destination; |
| 151 | + if (isSame(source, destination)) cout << 1 << endl; |
| 152 | + else cout << 0 << endl; |
| 153 | +} |
| 154 | +``` |
| 155 | + |
0 commit comments