-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathcourse_schedule_2.py
147 lines (129 loc) · 5.44 KB
/
course_schedule_2.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
import unittest
import collections
from typing import List
import enum
# 给你课程的依赖关系,求其中一种学完全部课程的学习顺序
# course_schedule_1 求的是可行性
# course_schedule_2 求的是其中一种拓扑排序
# 除了返回值,1和2几乎一样
class Solution(unittest.TestCase):
@staticmethod
def course_schedule_1(n: int, prerequisites: List[List[int]]) -> bool:
indegree = [0] * n
# 相当于邻接表
next_node: List[List[int]] = [[] for _ in range(n)]
# Build Graph
for node, prev in prerequisites:
indegree[node] += 1
next_node[prev].append(node)
# Find zero-indegree as queue init value
zero_indegree_q = collections.deque()
for node in range(n):
if indegree[node] == 0:
zero_indegree_q.append(node)
topological_order = []
while zero_indegree_q:
node = zero_indegree_q.popleft()
topological_order.append(node)
for next in next_node[node]:
indegree[next] -= 1
if indegree[next] == 0:
zero_indegree_q.append(next)
return len(topological_order) == n
def course_order(courses_count: int, prerequisites: List[List[int]]) -> List[int]:
"""
prerequisites可以理解成一种简单的有向图邻接表(adjacency list)
Input: 4, [[1,0],[2,0],[3,1],[3,2]]
Output: [0,1,2,3] or [0,2,1,3]
"""
# index: course_id, value: indegree's courses id
courses_indegree = []
# index: course_id, value: next_course require current course_id
next_courses: List[List[int]] = []
for _ in range(courses_count):
courses_indegree.append(0)
next_courses.append([])
for course_id, pre_course_id in prerequisites:
courses_indegree[course_id] += 1
next_courses[pre_course_id].append(course_id)
zero_indegree_queue = collections.deque()
for course_id in range(courses_count):
if courses_indegree[course_id] == 0:
zero_indegree_queue.append(course_id)
learn_order = []
learn_order_len = 0
while zero_indegree_queue:
course_id = zero_indegree_queue.popleft()
learn_order.append(course_id)
learn_order_len += 1
for next_course_id in next_courses[course_id]:
courses_indegree[next_course_id] -= 1
if courses_indegree[next_course_id] == 0:
zero_indegree_queue.append(next_course_id)
# 课程表1这题求能否学完,所以仅仅是返回值不同
# 图中有死循环,不能完整地学完课程
if learn_order_len != courses_count:
return []
return learn_order
# DFS过程中节点的状态(所以拓扑排序还是BFS简单)
class State(enum.Enum):
"""
对于图中的任意一个节点,它在搜索的过程中有三种状态,即:
「未搜索」:我们还没有搜索到这个节点;
「搜索中」:我们搜索过这个节点,但还没有回溯到该节点,即该节点还没有入栈,还有相邻的节点没有搜索完成
「已完成」:我们搜索过并且回溯过这个节点,即该节点已经入栈,并且所有该节点的相邻节点都出现在栈的更底部的位置,满足拓扑排序的要求
"""
UNVISITED = 0
SEARCHING = 1
VISITED = 2
def dfs(
course_id: int,
next_courses: List[List[int]],
visited: List[State],
result: List[int],
) -> bool:
"""
@return: is_circular_dependency
"""
visited[course_id] = State.SEARCHING
for next_course_id in next_courses[course_id]:
if visited[next_course_id] == State.UNVISITED:
if dfs(next_course_id, next_courses, visited, result):
return True
elif visited[next_course_id] == State.SEARCHING:
# 出现循环依赖
# is_circular_dependency = True
return True
visited[course_id] = State.VISITED
result.insert(0, course_id)
return False
def dfs_entrance(courses_count: int, prerequisites: List[List[int]]) -> List[int]:
visited: List[State] = []
# index: course_id, value: next_course require current course_id
next_courses: List[List[int]] = []
for _ in range(courses_count):
visited.append(State.UNVISITED)
next_courses.append([])
for course_id, pre_course_id in prerequisites:
next_courses[pre_course_id].append(course_id)
result = []
# is_circular_dependency = False
for course_id in range(courses_count):
if visited[course_id] == State.UNVISITED:
if dfs(course_id, next_courses, visited, result):
return []
return result
class Testing(unittest.TestCase):
TEST_CASES = [
(2, [[1, 0]], [0, 1]),
(4, [[1, 0], [2, 0], [3, 1], [3, 2]], [0, 1, 2, 3])
]
def test_course_order(self):
for courses_count, prerequisites, learn_order in self.TEST_CASES:
self.assertCountEqual(learn_order, course_order(courses_count, prerequisites))
def test_dfs(self):
for courses_count, prerequisites, learn_order in self.TEST_CASES:
# two list contain the same elements but order is different
# a and b have the same elements in the same number, regardless of their order
# https://stackoverflow.com/questions/12813633/how-to-assert-two-list-contain-the-same-elements-in-python
self.assertCountEqual(learn_order, dfs_entrance(courses_count, prerequisites))