大家好,我是挨打的阿木木,爱好算法的前端摸鱼老。最近会频繁给大家分享我刷算法题过程中的思路和心得。如果你也是想提高逼格的摸鱼老,欢迎关注我,一起学习。
题目
685. 冗余连接 II
在本问题中,有根树指满足以下条件的 有向 图。该树只有一个根节点,所有其他节点都是该根节点的后继。该树除了根节点之外的每一个节点都有且只有一个父节点,而根节点没有父节点。
输入一个有向图,该图由一个有着 n 个节点(节点值不重复,从 1 到 n)的树及一条附加的有向边构成。附加的边包含在 1 到 n 中的两个不同顶点间,这条附加的边不属于树中已存在的边。
结果图是一个以边组成的二维数组 edges 。 每个元素是一对 [ui, vi],用以表示 有向 图中连接顶点 ui 和顶点 vi 的边,其中 ui 是 vi 的一个父节点。
返回一条能删除的边,使得剩下的图是有 n 个节点的有根树。若有多个答案,返回最后出现在给定二维数组的答案。
示例 1:
输入: edges = [[1,2],[1,3],[2,3]]
输出: [2,3]
示例 2:
输入: edges = [[1,2],[2,3],[3,4],[4,1],[1,5]]
输出: [4,1]
提示:
n == edges.length3 <= n <= 1000edges[i].length == 21 <= ui, vi <= n
思路
- 这道题目和之前做过的冗余连接很相似,不过这道题目多了个难点,就是有向的树结构,这就意味着我们每个节点最多只能有一个父节点了;
- 我们可以先经过一轮判断,看看存不存在同时拥有两个父节点的节点,如果存在,只需要再它的两条链接中删除一条;
- 那么我们该删除哪一条呢,那么我们可以分类讨论,尽可能删除后面那条,除非删除了后面那条后整个结构无法全部串通,这种情况下才删第一条;
- 那么我们可以直接删除第二条,判断整个结构链接后是否全部串通,如果是的话直接返回我们删除掉的那个节点;
- 如果不是的话,说明第二条不能删,只能删第一条;
- 如果全部遍历完,发现并没有任何节点存在两个父节点,那么我们走正常删除逻辑即可。也就是把最晚出现的一条已经打通了的多余链接干掉。
实现
/**
* @param {number[][]} edges
* @return {number[]}
*/
var findRedundantDirectedConnection = function(edges) {
const n = edges.length;
const uf = new UnionFind(n);
let map = new Map();
let set = new Set();
let index1 = index2 = -1;
for (let i = 0; i < n; i++) {
const [ a, b ] = edges[i];
// 判断是否有两个父节点的,有的话说明是多余的
if (map.has(b)) {
index1 = map.get(b);
index2 = i;
break;
} else {
map.set(b, i);
set.add(a);
}
set.add(b);
}
// 如果存在特殊节点,那么在两个特殊节点的边中选择一个
if (index1 > -1) {
const result = edges.splice(index2, 1);
return mergeNode(edges, uf) ? result[0] : edges[index1];
} else {
return judgeNode(edges, uf);
}
};
// 串接节点后判断是否全部链接成功
function mergeNode(edges, uf) {
const n = edges.length;
// 先做链接
for (let i = 0; i < n; i++) {
let [ a, b ] = edges[i];
uf.merge(a - 1, b - 1);
}
// 判断是否全连起来了
const parentNode = uf.find(0);
return uf.parent.every((item, index) => uf.find(index) === parentNode);
}
// 判断节点是否可以链接起来
function judgeNode(edges, uf) {
const n = edges.length;
// 没有特殊情况,正常判断删除一条边即可
for (let i = 0; i < n; i++) {
let [ a, b ] = edges[i];
if (uf.find(a - 1) === uf.find(b - 1)) {
return [ a, b ];
} else {
uf.merge(a - 1, b - 1);
}
}
}
class UnionFind {
constructor(n) {
// 一开始每个元素的父元素都是自己
this.parent = new Array(n).fill(0).map((item, index) => index);
}
// 找到元素的父元素
find(index) {
return this.parent[index] = this.parent[index] === index ? index : this.find(this.parent[index]);
}
// 把index2的父元素设置为index1的父元素
merge(index1, index2) {
this.parent[this.find(index2)] = this.find(index1);
}
}
看懂了的小伙伴可以点个关注、咱们下道题目见。如无意外以后文章都会以这种形式,有好的建议欢迎评论区留言。