发布于 2016-01-11 08:33:55 | 207 次阅读 | 评论: 0 | 来源: PHPERZ
Leetcode 在线编程网站
leetcode 是一个美国的在线编程网站,上面主要收集了各大IT公司的笔试面试题,对于应届毕业生找工作是一个不可多得的好帮手。
Given
n
nodes labeled from0
ton - 1
and a list of undirected edges (each edge is a pair of nodes), write a function to check whether these edges make up a valid tree.For example:
Given
n = 5
andedges = [[0, 1], [0, 2], [0, 3], [1, 4]]
, returntrue
.Given
n = 5
andedges = [[0, 1], [1, 2], [2, 3], [1, 3], [1, 4]]
, returnfalse
.Note: you can assume that no duplicate edges will appear in
edges
. Since all edges are undirected,[0, 1]
is the same as[1, 0]
and thus will not appear together inedges
.
这种验证有效的图都是一类题。无非就是几种方法,比如DFS。注意树有个特征就是一个child不能有两个parent, 根平时拓扑排序找环的的差别,当然由于这道题是无向图,不用考虑这方面。这题有几个要注意的地方,比如一个树只能有一个root,如果这个树存在多个root, 那肯定无效。
由于这道题是无向图,那么我们其实可以从任意一个点开始DFS,不用必须从root,因为无向图从任意一点都可以遍历到整块部分。添加edge的时候两个方向都添就可以了,另外就是还有个注意的地方就是如果在遍历neighbors的时候,自己parent不要遍历,否则直接返回false以为有环了。总结一句就是,这种无向图每个点遍历一次就可以了,否则就意味着有环。
如果这道题换成是有向图的话,解法见下文。
time: O(V + E), space: O(h)
public class Solution {
public boolean validTree(int n, int[][] edges) {
List<Integer>[] graph = new List[n];
for (int i = 0; i < n; i++) {
graph[i] = new ArrayList<Integer>();
}
for (int i = 0; i < edges.length; i++) {
graph[edges[i][0]].add(edges[i][1]);
graph[edges[i][1]].add(edges[i][0]);
}
boolean[] visited = new boolean[n];
if (!dfs(graph, 0, visited, -1)) {
return false;
}
// 检测是否存在多个root, 如是意味着图无效
for (int i = 0; i < n; i++) {
if (!visited[i]) {
return false;
}
}
return true;
}
private boolean dfs(List<Integer>[] graph, int i, boolean[] visited, int prev) {
// 发现环
if (visited[i]) {
return false;
}
visited[i] = true;
for (int num : graph[i]) {
if (prev != num && !dfs(graph, num, visited, i)) {
return false;
}
}
return true;
}
}
Given
n
nodes labeled from0
ton - 1
and a list of directed edges (each edge is a pair of nodes), write a function to check whether these edges make up a valid tree.For example:
Given
n = 5
andedges = [[0, 1], [0, 2], [0, 3], [1, 4]]
, returntrue
.Given
n = 5
andedges = [[0, 1], [1, 2], [2, 3], [1, 3], [1, 4]]
, returnfalse
.
Givenn = 5
andedges = [[0, 1], [1, 2], [3, 4]]
, returnfalse
.
有向图要注意的地方不仅是不能有多个root存在,与上题不同的是需要从root开始遍历才能慢慢遍历到整个图,其他方面基本跟上题一样,遍历过程中一个点只能visit一次,否则即是那种一个child多个parent情况。
time: O(V+E), space: O(h)
public class Solution {
public boolean validTree(int n, int[][] edges) {
List<Integer>[] graph = new List[n];
Set<Integer> inDegree = new HashSet<>();
for (int i = 0; i < n; i++) {
graph[i] = new ArrayList<Integer>();
}
for (int i = 0; i < edges.length; i++) {
graph[edges[i][0]].add(edges[i][1]);
inDegree.add(edges[i][1]);
}
int root = 0;
int count = 0;
for (int i = 0; i < n; i++) {
if (!inDegree.contains(i)) {
root = i;
count++;
}
}
if (count != 1)
return false;
boolean[] visited = new boolean[n];
if (!dfs(graph, root, visited)) {
return false;
}
return true;
}
private boolean dfs(List<Integer>[] graph, int i, boolean[] visited) {
// 发现环或一child多个parents
if (visited[i]) {
return false;
}
visited[i] = true;
for (int num : graph[i]) {
if (!dfs(graph, num, visited)) {
return false;
}
}
return true;
}
}