当前位置: 首页 > news >正文

三色标记法 判断有向图是否有环

文章目录

  • 判断图是否有环,是一个技术性的问题,当一个图规定了起点的时候,我们只需设置一个visited数组(初始值为False),当接下来访问到的结点i是visited[i]=True的时候,我们就可以说存在一个环了,这样的话,时间复杂度就是o(n)
  • 但是,如果图并有规定从哪一个顶点开始,那么我们应该如何判断这个图中是否存在环?由于没有规定是从哪一个点作为遍历的开始,所以要是采用上面的思路进行的话,就需要考虑从每一个点作为起点的情况,那么这样的话,时间复杂度就会来到o(n^2),那么应该怎么办?

下面介绍三色标记法,也就是将结点分为,未被访问、正在访问、已经访问过,这三种情况

三色标记法

  • 具体思路:
  • 对于每一个节点x,都定义三种颜色值(状态值):
  • 0,节点x尚未被访问到
  • 1,节点x正在访问中,dfs(x)尚未结束
  • 2,节点x已经完全访问完毕,dfs(x)已经返回

注意:只使用两种状态是不能处理是否存在环的

我们所说的,节点x正在访问中,是说我们正在递归处理节点x以及它的后续节点,dfs(x)尚未结束

  • 时间复杂度:每个节点只会被访问一次,每一条边也会被访问一次,所以时间复杂度是o(N+E)

在这里插入图片描述
在这里插入图片描述

  • 思路分析:直接套用三色标记法即可

Python思路

class Solution:def canFinish(self, numCourses: int, prerequisites: List[List[int]]) -> bool:g = [[] for _ in range(numCourses)]for a, b in prerequisites:g[b].append(a)colors = [0] * numCourses# 返回 True 表示找到了环def dfs(x: int) -> bool:colors[x] = 1  # x 正在访问中for y in g[x]:if colors[y] == 1 or colors[y] == 0 and dfs(y):return True  # 找到了环colors[x] = 2  # x 完全访问完毕return False  # 没有找到环for i, c in enumerate(colors):if c == 0 and dfs(i):return False  # 有环return True  # 没有环

C++思路

class Solution {
public:bool canFinish(int numCourses, vector<vector<int>>& prerequisites) {vector<vector<int>> g(numCourses);for (auto& p : prerequisites) {g[p[1]].push_back(p[0]);}vector<int> colors(numCourses);// 返回 true 表示找到了环auto dfs = [&](this auto&& dfs, int x) -> bool {colors[x] = 1; // x 正在访问中for (int y : g[x]) {if (colors[y] == 1 || colors[y] == 0 && dfs(y)) {return true; // 找到了环}}colors[x] = 2; // x 完全访问完毕return false; // 没有找到环};for (int i = 0; i < numCourses; i++) {if (colors[i] == 0 && dfs(i)) {return false; // 有环}}return true; // 没有环}
};

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

  • 思路分析:

python代码

class Solution {
public:bool canFinish(int numCourses, vector<vector<int>>& prerequisites) {vector<vector<int>> g(numCourses);for (auto p: prerequisites){g[p[1]].push_back(p[0]);}vector<int> colors(numCourses);auto dfs = [&](this auto && dfs,int x) -> bool{colors[x] = 1;for (int y : g[x]){if (colors[y] == 1 || colors[y] == 0 && dfs(y)){return true;}}colors[x] = 2;return false;};for (int i = 0; i < numCourses; i++){if (colors[i] == 0 && dfs(i)){return false;}}return true;}
};

相关文章:

  • Leetcode 3562. Maximum Profit from Trading Stocks with Discounts
  • CAD精简多段线顶点、优化、删除多余、重复顶点——CAD c#二次开发
  • 厚铜PCB线路板厂会如何处理质量问题?
  • 滑动窗口算法实时计算QPS:Java实现与原理分析
  • MAR:无需量化的掩码自回归图像生成模型
  • RuoYi前后端分离框架集成UEditorPlus富文本编辑器
  • 嵌入式学习(基本操作)day1
  • 在麒麟系统(Kylin OS)上安装`geckodriver`
  • 认识微服务
  • DNS 详情 新增 DNS 自适应服务器 ip
  • 【部署】在离线服务器的docker容器下升级dify-import程序
  • leetcode 3559. Number of Ways to Assign Edge Weights II
  • 【Hive基础】01.数据模型、存储格式、排序方式
  • 内网映射有什么作用,如何实现内网的网络地址映射到公网连接?
  • 【图像处理基石】什么是色彩模式?
  • 倚光科技在二元衍射面加工技术上的革新:引领光学元件制造新方向​
  • 佰力博科技与您探讨铁电分析仪具有哪些测试功能
  • 游戏引擎学习第313天:回到 Z 层级的工作
  • 聊天室H5实时群聊聊天室全开源系统(源码下载)
  • Lines of Thought in Large Language Models
  • 页面设计论文/seo优化员
  • 阿里巴巴国际站入驻费用及条件/推广网页
  • 访问自己做的网站/百度投诉中心24人工
  • 怎样做网站的优化 排名/上海优化seo排名
  • 达县网站制作/优化关键词是什么意思
  • 做潮鞋的网站和平台/人工智能培训班收费标准