当前位置: 首页 > 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;}
};
http://www.xdnf.cn/news/659791.html

相关文章:

  • 高并发系统下Mutex锁、读写锁、线程重入锁的使用思考
  • 区块链DApp的开发技术方案
  • 04_redis之ZSet使用实例-积分榜
  • 如何提高 Python 代码质量
  • 数据安全与纵深访问控制:构建数字时代的安全防线
  • 三、Docker目录挂载、卷映射、网络
  • 量子-经典协同计算新路径:NISQ 时代混合算法对后量子密码学的适应性探索
  • Linux系统编程-DAY05
  • 华为OD机试真题——最长的顺子(2025B卷:100分)Java/python/JavaScript/C++/C语言/GO六种最佳实现
  • SOC-ESP32S3部分:14-错误处理
  • 【教学类-36-09】20250526动物面具描边(通义万相)对称图40张,根据图片长宽,自动旋转图片,最大化图片
  • vue3组合API-toRefs函数
  • Python 训练营打卡 Day 36
  • A2A协议(Agent-to-agent Protocol)学习
  • CentOS中安装Docker Compose
  • 【面试题】如何测试一个新增的服务端接口?
  • CSS闯关指南:从手写地狱到“类”积木之旅|得物技术
  • 嵌入式开发学习日志(linux系统编程--进程(2))Day28
  • TLS/PSK
  • vue3减少打包体积
  • C++:多重继承
  • 蓝桥杯b组c++赛道---数位dp
  • git 新建一个分支,怎么首次推到远程仓库
  • 计算机图形学:(四)欧拉角与四元数
  • 尚硅谷redis7 37 redis持久化之AOF简介
  • Unity---OSC(Open Sound Control)、TouchOSC Editor、创建布局
  • Java高频面试之并发编程-21
  • Linux `hostname` 命令深度解析与高阶应用指南
  • Linux中的SELinux
  • RPM之(1)基础使用