当前位置:   article > 正文

leetcode 684. 冗余链接 802. 找到最终的安全状态

leetcode 684

在本问题中, 树指的是一个连通且无环的无向图。

输入一个图,该图由一个有着N个节点 (节点值不重复1, 2, ..., N) 的树及一条附加的边构成。附加的边的两个顶点包含在1到N中间,这条附加的边不属于树中已存在的边。

结果图是一个以组成的二维数组。每一个的元素是一对[u, v] ,满足 u < v,表示连接顶点u 和v无向图的边。

返回一条可以删去的边,使得结果图是一个有着N个节点的树。如果有多个答案,则返回二维数组中最后出现的边。答案边 [u, v] 应满足相同的格式 u < v

示例 1:

输入: [[1,2], [1,3], [2,3]]
输出: [2,3]
解释: 给定的无向图为:
  1
 / \
2 - 3

示例 2:

输入: [[1,2], [2,3], [3,4], [1,4], [1,5]]
输出: [1,4]
解释: 给定的无向图为:
5 - 1 - 2
    |   |
    4 - 3

注意:

  • 输入的二维数组大小在 3 到 1000。
  • 二维数组中的整数在1到N之间,其中N是输入数组的大小。

 

代码

  1. class Solution {
  2. public class WeightedQuickUnionUF {
  3. private int[] parent;
  4. private int[] size;
  5. public WeightedQuickUnionUF(int n) {
  6. parent = new int[n];
  7. size = new int[n];
  8. for (int i = 0; i < n; i++) {
  9. parent[i] = i;
  10. size[i] = 1;
  11. }
  12. }
  13. public int find(int p) {
  14. while (p != parent[p])
  15. p = parent[p];
  16. return p;
  17. }
  18. public boolean connected(int p, int q) {
  19. return find(p) == find(q);
  20. }
  21. public void union(int p, int q) {
  22. int rootP = find(p);
  23. int rootQ = find(q);
  24. if (rootP == rootQ) return;
  25. if (size[rootP] < size[rootQ]) {
  26. parent[rootP] = rootQ;
  27. size[rootQ] += size[rootP];
  28. }
  29. else {
  30. parent[rootQ] = rootP;
  31. size[rootP] += size[rootQ];
  32. }
  33. }
  34. }
  35. public int[] findRedundantConnection(int[][] edges) {
  36. WeightedQuickUnionUF uf = new WeightedQuickUnionUF(edges.length);
  37. int[] result = new int[2];
  38. for(int i=0;i<edges.length;i++){
  39. int vertex1 = edges[i][0]-1;
  40. int vertex2 = edges[i][1]-1;
  41. if(!uf.connected(vertex1,vertex2)){
  42. uf.union(vertex1,vertex2);
  43. }
  44. else{
  45. result[0]=vertex1+1;
  46. result[1]=vertex2+1;
  47. break;
  48. }
  49. }
  50. return result;
  51. }
  52. }

在有向图中, 我们从某个节点和每个转向处开始, 沿着图的有向边走。 如果我们到达的节点是终点 (即它没有连出的有向边), 我们停止。

现在, 如果我们最后能走到终点,那么我们的起始节点是最终安全的。 更具体地说, 存在一个自然数 K,  无论选择从哪里开始行走, 我们走了不到 K 步后必能停止在一个终点。

哪些节点最终是安全的? 结果返回一个有序的数组。

该有向图有 N 个节点,标签为 0, 1, ..., N-1, 其中 N 是 graph 的节点数.  图以以下的形式给出: graph[i] 是节点 j的一个列表,满足 (i, j) 是图的一条有向边。

示例:
输入:graph = [[1,2],[2,3],[5],[0],[5],[],[]]
输出:[2,4,5,6]
这里是上图的示意图。

Illustration of graph

提示:

  • graph 节点数不超过 10000.
  • 图的边数不会超过 32000.
  • 每个 graph[i] 被排序为不同的整数列表, 在区间 [0, graph.length - 1] 中选取。

代码

  1. class Solution {
  2. public boolean hasCircle = false;
  3. public void dfs(int i, int[][]graph, int[] visited, int[] onStack){
  4. visited[i]=1;
  5. onStack[i]=1;
  6. for(int j:graph[i]){
  7. if(visited[j]==0){
  8. dfs(j,graph,visited,onStack);
  9. }else if(onStack[j]==1){
  10. hasCircle = true;
  11. break;
  12. }
  13. }
  14. onStack[i]=0;
  15. }
  16. public List<Integer> eventualSafeNodes(int[][] graph) {
  17. List<Integer> results = new ArrayList<Integer>();
  18. for(int i=0;i<graph.length;i++){
  19. int[] visited = new int[graph.length];
  20. int[] onStack = new int[graph.length];
  21. dfs(i,graph,visited,onStack);
  22. if(!hasCircle){
  23. results.add(i);
  24. }
  25. hasCircle = false;
  26. }
  27. return results;
  28. }
  29. }

 

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/weixin_40725706/article/detail/995309
推荐阅读
相关标签
  

闽ICP备14008679号