LeetCode[207]课程表

难度:Medium

题目:

你这个学期必须选修 numCourses 门课程,记为 0numCourses - 1

在选修某些课程之前需要一些先修课程。 先修课程按数组 prerequisites 给出,其中 prerequisites[i] = [ai, bi] ,表示如果要学习课程 ai必须 先学习课程 bi

  • 例如,先修课程对 [0, 1] 表示:想要学习课程 0 ,你需要先完成课程 1

请你判断是否可能完成所有课程的学习?如果可以,返回 true ;否则,返回 false


示例 1:

复制代码
输入:numCourses = 2, prerequisites = [[1,0]]
输出:true
解释:总共有 2 门课程。学习课程 1 之前,你需要完成课程 0 。这是可能的。

示例 2:

复制代码
输入:numCourses = 2, prerequisites = [[1,0],[0,1]]
输出:false
解释:总共有 2 门课程。学习课程 1 之前,你需要先完成​课程 0 ;并且学习课程 0 之前,你还应先完成课程 1 。这是不可能的。

提示:

  • 1 <= numCourses <= 2000
  • 0 <= prerequisites.length <= 5000
  • prerequisites[i].length == 2
  • 0 <= ai, bi < numCourses
  • prerequisites[i] 中的所有课程对 互不相同

Related Topics

  • 深度优先搜索
  • 广度优先搜索
  • 拓扑排序

重点!!!解题思路

第一步:

明确解题手段:仔细读题,可发现要先完成前置课程,必须先完成后置课程,那么后置课程其实可以看做是前置课程的一个入度(indegree),只有当一个节点入度为0时,它才可以被学习,理解到这里就可以发现此题可使用拓扑排序来解决。

第二步:

拓扑排序思路:1.把原数组关系看作是一个图

2.需要一个能记录每个节点的入度的数组

3.需要一个二维集合来记录每个0入度节点对应的子节点

4.需要一个队列来将入度为0的节点添加到队列中去,每次出队的时候找到它们对应的子节点,并将子节点的入度-1,当子节点的入度为0就添加到队列中,继续判断

5.最后当出队数量与原数量一致,那么即为正确答案

源码+讲解:

java 复制代码
class Solution {
    public boolean canFinish(int numCourses, int[][] prerequisites) {
        int[] indeg=new int[numCourses];  //这个数组:入度统计
        List<List<Integer>> g=new ArrayList<>();  //二维集合: 统计每个入度为0节点对应的子节点
        for (int i=0;i<numCourses;i++){  //首先初始化二维集合
            g.add(new ArrayList<>());
        }
        Queue<Integer> q=new ArrayDeque<>();  //创建一个队列用于计数操作
        for (int[] prerequisite : prerequisites) {
            indeg[prerequisite[0]]++;  //遍历给定二维数组,前置节点入度+1
            g.get(prerequisite[1]).add(prerequisite[0]);  //后置节点对应一个前置节点
        }
        for (int i=0;i<indeg.length;i++){  //将入度为0的节点添加到队列中去
            if (indeg[i]==0) q.offer(i);
        }
        while (!q.isEmpty()){  
            int pre = q.poll();  //每次弹出的都是一个入度为0的前置节点
            numCourses--;  //每弹出一个,相当于总节点数少一个,当numCourses为0时,才是答案
            for (Integer cur : g.get(pre)) {  //根据前置节点取出所有后置节点,并将每个后置节点入度-1,直到减为0
                if (--indeg[cur]==0) q.offer(cur);
            }
        }
        return numCourses==0;
    }
}

运行结果:

如果您还有什么疑问或解答有问题,可在下方评论,我会及时回复。

系列持续更新中,点个订阅吧,喜欢练习算法那就点个攒吧

相关推荐
EterNity_TiMe_11 分钟前
【论文复现】(CLIP)文本也能和图像配对
python·学习·算法·性能优化·数据分析·clip
机器学习之心21 分钟前
一区北方苍鹰算法优化+创新改进Transformer!NGO-Transformer-LSTM多变量回归预测
算法·lstm·transformer·北方苍鹰算法优化·多变量回归预测·ngo-transformer
yyt_cdeyyds32 分钟前
FIFO和LRU算法实现操作系统中主存管理
算法
alphaTao1 小时前
LeetCode 每日一题 2024/11/18-2024/11/24
算法·leetcode
kitesxian1 小时前
Leetcode448. 找到所有数组中消失的数字(HOT100)+Leetcode139. 单词拆分(HOT100)
数据结构·算法·leetcode
VertexGeek2 小时前
Rust学习(八):异常处理和宏编程:
学习·算法·rust
石小石Orz2 小时前
Three.js + AI:AI 算法生成 3D 萤火虫飞舞效果~
javascript·人工智能·算法
jiao_mrswang3 小时前
leetcode-18-四数之和
算法·leetcode·职场和发展
qystca3 小时前
洛谷 B3637 最长上升子序列 C语言 记忆化搜索->‘正序‘dp
c语言·开发语言·算法
薯条不要番茄酱3 小时前
数据结构-8.Java. 七大排序算法(中篇)
java·开发语言·数据结构·后端·算法·排序算法·intellij-idea