Detect Cycle in Directed Graph 有向图找环
Given n nodes labeled from 0 to n - 1 and a list of directed edges (each edge is a pair of nodes), write a function to check whether the graph contains a cycle. if edges = [0, 1], [1, 2], [0, 2]], it means 0 -> 1, 1 ->2, 0 -> 2. edges[i is parent, edgesi is child.
For example:
Given n = 3 and edges = [[0, 1], [1, 2], [0, 2], return true.
Given n = 3 and edges = [[0, 1], [1, 2], [2, 0], return false.
黑白灰DFS大法
复杂度
O( V + E ) 时间 O(V) 空间
思路
什么是有向图有环:只要从a可以到a,经过的边的个数大于等于1
做法:
维护三个点的集合:
白点集合,里面放还没explore的点
灰点集合,里面放正在explore的点,当前的灰点们表示一条正在explore的路径,这个路径上的每个点都是灰的
黑点集合,里面放已经explore的点且这些点不构成环
如果我将要的访问的点是黑点,我是否需要访问他?
答:不需要,一个点是黑的表示这个点的所有孩子(邻居)都explore过了,且没发现环,且这个点的所有孩子必定也全是黑的。
何时把一个点由灰变黑?
答:当这个点的所有孩子都访问完(都已变黑)了且没有发现环
如果我将要访问的点是个灰点,说明什么?
答:发现了环。假设explore到了cur点,cur点为灰色,此时所有其他的灰色点必定都是我的祖先,因为他们都是当前explore的路径上的点,cur在最战线的最前方explore,如果cur点在explore的时候发现自己的的孩子(邻居)有一个灰色,表示下面这个点即是我的祖先也是我的孩子,说明从cur可以走到cur自己,即出现了环。
注意
无向图和有向图很不一样,不是一回事
代码
核心代码:
public boolean hasCycleDirectedGraph(int n, int[][] edges) {//前指后
HashSet<Integer> black = new HashSet<Integer>();
HashSet<Integer> white = new HashSet<Integer>();
HashSet<Integer> gray = new HashSet<Integer>();
List<List<Integer>> adjList = new ArrayList<List<Integer>>();
for (int i = 0; i < n; ++i) {
white.add(new Integer(i));
adjList.add(new ArrayList<Integer>());
}
for (int[] edge: edges) {
adjList.get(edge[0]).add(new Integer(edge[1]));
}
for (int i = 0; i < n; i++) {
if (white.contains(i)) {
if (hasCycle(i, white, gray, black, adjList))
return true;
}
}
return false;
}
private boolean hasCycle(Integer vertex, HashSet<Integer> white, HashSet<Integer> gray, HashSet<Integer> black, List<List<Integer>> adjList) {
white.remove(vertex);
gray.add(vertex); // current vertex is being visited
for (Integer succ: adjList.get(vertex)) { // successors of current vertex
if (white.contains(succ)) {
if (hasCycle(succ, white, gray, black, adjList))
return true;
}
else if (gray.contains(succ)) {
return true;
}
else if (black.contains(succ)) {
continue;
}
}
gray.remove(vertex);
black.add(vertex);
return false;
}
测试代码:
public class Solution {
public static void main(String[] args) {
Solution so = new Solution();
int[][] edges = {{0, 1}, {1, 2}, {2, 0}};
boolean result = so.hasCycleDirectedGraph(3, edges);
System.out.println(result);
}
}
Detect Cycle in Undirected Graph 无向图找环
Given n nodes labeled from 0 to n - 1 and a list of undirected edges (each edge is a pair of nodes), write a function to check whether the graph contains a cycle.
For example:
Given n = 5 and edges = [[0, 1], [0, 2], [0, 3], [1, 4]], return false.
Given n = 5 and edges = [[0, 1], [1, 2], [2, 3], [1, 3], [1, 4]], return true.
并查集大法
复杂度
O( V + E ) 时间 O(V) 空间
思路
什么是无向图有环:只要从a可以到a,路径中每个边只用一次
数据结构:
并查集:
规定集合(即一个连通分量)应该满足的property:里面的任意两点有且只有一条路径可达。
最开始的时候n个点有n个连通分量,即n个集合。
做法:
想象一开始这个图只有顶点,没有边,我们来一条一条的添加边。
每遇到一条边,判断这边的两个端点是否在同一个集合里?
在的话,表示有环:因为两个点在一个集合里就表示这两个点已经有一条路径了,现在再加一条路径,必然构成环。
不在的话,表示不构成环,我们应该合并这两个集合:因为加上这条边,两个集合就被连起来了,合并成了一个集合
注意
如果想要复杂度为O(V+E),必须用路径压缩。
并查集写法参考
注意union(p,q)方法:用find()找p,q的老大哥,合并的是p,q的老大哥。
代码
UnionFind Utility(非常intuitive,应该练习在5分钟内写完):
class UnionFind {
private int[] father;
private int count;
public UnionFind(int n) {
father = new int[n];
count = n;
for (int i = 0; i < n; i++){
father[i] = i;
}
}
public int count() {
return this.count;
}
public int find(int p) {
int root = father[p];
while (root != father[root])
root = father[root];
//as long as we get here, root is the final dad
while (p != root) {
int tmp = father[p];
father[p] = root;
p = tmp;
}
return root;
}
public void union(int p, int q) {
int fatherP = find(p);
int fatherQ = find(q);
if (fatherP != fatherQ) {
father[fatherP] = fatherQ;
count--;
}
}
}
主程序
public class Solution {
public boolean validTree(int n, int[][] edges) {
UnionFind uf = new UnionFind(n);
for (int[] edge : edges){
int p = edge[0];
int q = edge[1];
if (uf.find(p) == uf.find(q))
return false;
else
uf.union(p,q);
}
return uf.count() == 1;
}
}
DFS/BFS法
复杂度
O( V + E ) 时间 O(V) 空间
思路
无向图找环和有向图找环本质上完全不同。
有向图找环需要三种颜色。
无向图找环只需要两种颜色,就是访问过的和没访问的。
dfs过程中如果碰到访问过的节点(当然这个节点不能是来时的节点),就是有环。
注意
代码
public class Solution {
public boolean validTree(int n, int[][] edges) {
HashSet<Integer> visited = new HashSet<Integer>();
List<List<Integer>> adjList = new ArrayList<List<Integer>>();
for (int i=0; i<n; ++i)
adjList.add(new ArrayList<Integer>());
for (int[] edge: edges) {
adjList.get(edge[0]).add(edge[1]);
adjList.get(edge[1]).add(edge[0]);
}
if (hasCycle(-1, 0, visited, adjList)) // has cycle?
return false;
if (visited.size() != n) // is all connected?
return false;
return true;
}
private boolean hasCycle(Integer pred, Integer vertex, HashSet<Integer> visited, List<List<Integer>> adjList) {
visited.add(vertex); // current vertex is being visited
for (Integer succ: adjList.get(vertex)) { // successors of current vertex
if (!succ.equals(pred)) { // exclude current vertex's predecessor
if (visited.contains(succ)) {
return true; // back edge/loop detected!
}
else {
if (hasCycle(vertex, succ, visited, adjList)) {
return true;
}
}
}
}
return false;
}
}