美文网首页算法
210. 课程表 II

210. 课程表 II

作者: 红树_ | 来源:发表于2023-05-28 18:34 被阅读0次

参考210. 课程表 II

题目

现在你总共有 numCourses 门课需要选,记为 0numCourses - 1。给你一个数组 prerequisites ,其中 prerequisites[i] = [ai, bi] ,表示在选修课程 ai必须 先选修 bi

  • 例如,想要学习课程 0 ,你需要先完成课程 1 ,我们用一个匹配来表示:[0,1]

返回你为了学完所有课程所安排的学习顺序。可能会有多个正确的顺序,你只要返回 任意一种 就可以了。如果不可能完成所有课程,返回 一个空数组

输入:numCourses = 2, prerequisites = [[1,0]]
输出:[0,1]
解释:总共有 2 门课程。要学习课程 1,你需要先完成课程 0。因此,正确的课程顺序为 [0,1] 。
输入:numCourses = 4, prerequisites = [[1,0],[2,0],[3,1],[3,2]]
输出:[0,2,1,3]
解释:总共有 4 门课程。要学习课程 3,你应该先完成课程 1 和课程 2。并且课程 1 和课程 2 都应该排在课程 0 之后。
因此,一个正确的课程顺序是 [0,1,2,3] 。另一个正确的排序是 [0,2,1,3] 。
输入:numCourses = 1, prerequisites = []
输出:[0]

解题思路

  • 拓扑排序 + 广度优先搜索:构造有向图,使用bfs+入度,也可以使用dfs + 状态标记,搜索时倒序赋值结果。

拓扑排序 + 广度优先搜索

class Solution {
    public int[] findOrder(int n, int[][] prerequisites) {
        //构造图并初始化 list或hashmap或数组
        List<Integer>[] graph = new ArrayList[n];
        for(int i = 0; i < n; i++) {
            graph[i] = new ArrayList<>();
        }
        //同时初始化入度
        int[] indeg = new int[n];
        for(int[] pre : prerequisites) {
            graph[pre[1]].add(pre[0]);
            indeg[pre[0]] += 1;
        }
        //初始化结果
        int[] ans = new int[n];
        int index = 0;
        //初始化bfs 拓扑排序搜索节点
        Deque<Integer> q = new ArrayDeque<>();
        for(int i = 0; i < n; i++) {
            if(indeg[i] == 0) q.add(i);//入度为零入栈
        }
        while(q.size() > 0) {
            int i = q.poll();
            ans[index++] = i;
            //处理相邻节点
            for(int j : graph[i]) if(--indeg[j] == 0) q.add(j);
        }
        //根据下标和结果长度比较判断是否有环
        return  index == n ? ans : new int[]{};
    }
}

复杂度分析

  • 时间复杂度:O(n+m),其中 n 为课程数,m 为先修课程的要求数,T(n+m+n+m) = O(n+m)
  • 空间复杂度:O(n+m)

相关文章

网友评论

    本文标题:210. 课程表 II

    本文链接:https://www.haomeiwen.com/subject/skpdedtx.html