leetcode 2360 图中最长的环 题解

打印 上一主题 下一主题

主题 1387|帖子 1387|积分 4161

题面

给定一个有向图,每个点出度最大为一,现在问你图中最长的环的长度是多少,如果没有环输出 -1,                                   1                         ≤                         n                         ≤                         1                                   0                            5                                       1 \le n \le 10^5                  1≤n≤105。
题面
解题思路

我们直接说结论,我们从任意一个点出发,用一个数组存下来到达每一个点的最短步数,如果发现一个点之前已经被遍历过了,那么这个点一定是一个环的出发点和尽头,环的长度是当前的步数减去这个点的最短步数。
下面稍微证明一下,我们来看标题中给出的条件,每一个点最多只有一条出边,那么这个条件告诉了我们两件事变:

  • 边数比力小;
  • 图中不存在以下情况:

所以当我们遍历的时间,全部颠末的点和边形成的一定是一个链型的布局。当发现有一个点重复遍历的时间,一定是返回到了链上前面的某一个点,形成了一个环,我们统计一下答案就好了。
除此之外我们要注意,标题没有保证任意两点之间可以相互到达,当我们遍历一次之后我们还要查抄还有没有别的点我们没有遍历。
任意两次遍历是互不影响的,大概说一个环只可能出现在详细的一次遍历当中,所以当我们完成一次遍历的时间,直接把这次的点全部都删了就行,否则复杂度就上来了。
代码

  1. class Solution {
  2. public:
  3.     int ans = -1; // 最终答案
  4.     vector<int> dis; // 存最短步数
  5.     void dfs(int x, int D, vector<int>& edges) // D 表示当前步数
  6.     {
  7.         if (dis[x] != 0) // 如果发现之前被遍历过了,说明可能存在环
  8.         {
  9.             if (dis[x] == -1) // -1 说明是之前的遍历,不用管
  10.                 return;
  11.             ans = max(ans, D - dis[x]); // 否则就是存在环,记录答案
  12.             return;
  13.         }
  14.         dis[x] = D; // 维护 dis
  15.         if (edges[x] != -1)
  16.             dfs(edges[x], D + 1, edges); // 最多只有一个出度,就不用 for 循环了
  17.         dis[x] = -1; // 遍历完成开始回溯了,直接把这个点删掉
  18.     }
  19.     int longestCycle(vector<int>& edges) {
  20.         const int N = edges.size();
  21.         dis.resize(100005, 0);
  22.         for (int i = 0; i < N; i++) // 记得要遍历所有的点
  23.             if (dis[i] == 0)
  24.                 dfs(i, 1, edges);
  25.         return ans;
  26.     }
  27. };
复制代码









本人本领有限,如有不当之处敬请指教!

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

正序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

张裕

论坛元老
这个人很懒什么都没写!
快速回复 返回顶部 返回列表