class Solution {
public:
vector<int> findOrder(int numCourses, vector<vector<int>>& prerequisites) {
}
};
剑指 Offer II 113. 课程顺序
现在总共有 numCourses
门课需要选,记为 0
到 numCourses-1
。
给定一个数组 prerequisites
,它的每一个元素 prerequisites[i]
表示两门课程之间的先修顺序。 例如 prerequisites[i] = [ai, bi]
表示想要学习课程 ai
,需要先完成课程 bi
。
请根据给出的总课程数 numCourses
和表示先修顺序的 prerequisites
得出一个可行的修课序列。
可能会有多个正确的顺序,只要任意返回一种就可以了。如果不可能完成所有课程,返回一个空数组。
示例 1:
输入: numCourses = 2, prerequisites = [[1,0]] 输出:[0,1]
解释: 总共有 2 门课程。要学习课程 1,你需要先完成课程 0。因此,正确的课程顺序为[0,1] 。
示例 2:
输入: numCourses = 4, prerequisites = [[1,0],[2,0],[3,1],[3,2]] 输出:[0,1,2,3] or [0,2,1,3]
解释: 总共有 4 门课程。要学习课程 3,你应该先完成课程 1 和课程 2。并且课程 1 和课程 2 都应该排在课程 0 之后。 因此,一个正确的课程顺序是[0,1,2,3]
。另一个正确的排序是[0,2,1,3]
。
示例 3:
输入: numCourses = 1, prerequisites = []
输出: [0]
解释: 总共 1 门课,直接修第一门课就可。
提示:
1 <= numCourses <= 2000
0 <= prerequisites.length <= numCourses * (numCourses - 1)
prerequisites[i].length == 2
0 <= ai, bi < numCourses
ai != bi
prerequisites
中不存在重复元素
注意:本题与主站 210 题相同:https://leetcode.cn/problems/course-schedule-ii/
原站题解
golang 解法, 执行用时: 16 ms, 内存消耗: 6.4 MB, 提交时间: 2022-11-22 10:55:27
func findOrder(numCourses int, prerequisites [][]int) []int { var ( edges = make([][]int, numCourses) visited = make([]int, numCourses) result []int valid bool = true dfs func(u int) ) dfs = func(u int) { visited[u] = 1 for _, v := range edges[u] { if visited[v] == 0 { dfs(v) if !valid { return } } else if visited[v] == 1 { valid = false return } } visited[u] = 2 result = append(result, u) } for _, info := range prerequisites { edges[info[1]] = append(edges[info[1]], info[0]) } for i := 0; i < numCourses && valid; i++ { if visited[i] == 0 { dfs(i) } } if !valid { return []int{} } for i := 0; i < len(result)/2; i ++ { result[i], result[numCourses-i-1] = result[numCourses-i-1], result[i] } return result }
golang 解法, 执行用时: 16 ms, 内存消耗: 6 MB, 提交时间: 2022-11-22 10:55:00
func findOrder(numCourses int, prerequisites [][]int) []int { var ( edges = make([][]int, numCourses) indeg = make([]int, numCourses) result []int ) for _, info := range prerequisites { edges[info[1]] = append(edges[info[1]], info[0]) indeg[info[0]]++ } q := []int{} for i := 0; i < numCourses; i++ { if indeg[i] == 0 { q = append(q, i) } } for len(q) > 0 { u := q[0] q = q[1:] result = append(result, u) for _, v := range edges[u] { indeg[v]-- if indeg[v] == 0 { q = append(q, v) } } } if len(result) != numCourses { return []int{} } return result }
python3 解法, 执行用时: 52 ms, 内存消耗: 15.9 MB, 提交时间: 2022-11-22 10:54:45
class Solution: def findOrder(self, numCourses: int, prerequisites: List[List[int]]) -> List[int]: # 存储有向图 edges = collections.defaultdict(list) # 存储每个节点的入度 indeg = [0] * numCourses # 存储答案 result = list() for info in prerequisites: edges[info[1]].append(info[0]) indeg[info[0]] += 1 # 将所有入度为 0 的节点放入队列中 q = collections.deque([u for u in range(numCourses) if indeg[u] == 0]) while q: # 从队首取出一个节点 u = q.popleft() # 放入答案中 result.append(u) for v in edges[u]: indeg[v] -= 1 # 如果相邻节点 v 的入度为 0,就可以选 v 对应的课程了 if indeg[v] == 0: q.append(v) if len(result) != numCourses: result = list() return result
python3 解法, 执行用时: 36 ms, 内存消耗: 17.9 MB, 提交时间: 2022-11-22 10:53:32
class Solution: def findOrder(self, numCourses: int, prerequisites: List[List[int]]) -> List[int]: # 存储有向图 edges = collections.defaultdict(list) # 标记每个节点的状态:0=未搜索,1=搜索中,2=已完成 visited = [0] * numCourses # 用数组来模拟栈,下标 0 为栈底,n-1 为栈顶 result = list() # 判断有向图中是否有环 valid = True for info in prerequisites: edges[info[1]].append(info[0]) def dfs(u: int): nonlocal valid # 将节点标记为「搜索中」 visited[u] = 1 # 搜索其相邻节点 # 只要发现有环,立刻停止搜索 for v in edges[u]: # 如果「未搜索」那么搜索相邻节点 if visited[v] == 0: dfs(v) if not valid: return # 如果「搜索中」说明找到了环 elif visited[v] == 1: valid = False return # 将节点标记为「已完成」 visited[u] = 2 # 将节点入栈 result.append(u) # 每次挑选一个「未搜索」的节点,开始进行深度优先搜索 for i in range(numCourses): if valid and not visited[i]: dfs(i) if not valid: return list() # 如果没有环,那么就有拓扑排序 # 注意下标 0 为栈底,因此需要将数组反序输出 return result[::-1]