207. 课程表

题目描述

你这个学期必须选修 numCourses 门课程,记为 0 到 numCourses - 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 <= 105
0 <= prerequisites.length <= 5000
prerequisites[i].length == 2
0 <= ai, bi < numCourses
prerequisites[i] 中的所有课程对 互不相同
通过次数254,578提交次数472,004


做题思路

广度优先搜索

  • 要是往底层深究的话 你需要了解 图的相关知识 (有向无环图,有向有环图, 入度, 出度)

  • 但是为了方便讲解本题,我简单明了的将图的知识转换为方便大家理解的例子

  • 我是定义了一个集合数组,一个数组,一个队列

  • 一个个的来说各个代表什么意思

  • List[] adjs;

    我把这个集合数组的作用概括为 “谁需要我”

    我把下标看成被别人需要的课程编号,里面的值看成 需要我的课程编号

  • int[] indegress;

我把这个数组的作用概括为 “我依赖别人的个数”

下标我看成需要依赖别人的课程编号,值我看成 “我依赖别人的个数”

  • Queue canStudy

    这个队列里面的值我放的是 可以学习的课程的编号

    这个的值可以从 indegress数组里面去找 值 为0的下标就是可以现在已经可以学习的课程的编号

  • 然后大概的实现思路我说一下:每次从队列里面"拿出来"一个现在就可以学习的课程的编号,然后再去adjs里面去看需要我的课程id都有谁,依次取出来,到indegress数组里面 “消消乐”

    直到canStudy队列里面的元素为空(可以学习的课程为0的时候)就可以结束了

  • 最后用我们已经学习过的课程和题目中给出的我们需要学习的课程对比即可返回结果

代码实现

class Solution {
    public boolean canFinish(int numCourses, int[][] prerequisites) {
        
        //谁需要我!!!!
        List<Integer>[] adjs =new List[numCourses];

        //我需要别人的个数!!!
        int[] indegress=new int[numCourses];


        for(int i=0;i<numCourses;i++){
            adjs[i]=new ArrayList<>();
        }


        for(int i=0;i<prerequisites.length;i++){
            adjs[prerequisites[i][1]].add(prerequisites[i][0]);
            indegress[prerequisites[i][0]]++;
        }

        //判断 "如果我需要别人的个数为0的话就加入这个集合中:表明现在可以学习了"
        Queue<Integer> canStudy=new LinkedList<>();
        
        
        for(int i=0;i<indegress.length;i++){
            if(indegress[i]==0){
                canStudy.add(i);
            }
        }

        int haveStudyCource=0;
        while(!canStudy.isEmpty()){
            int canStudyCource=canStudy.remove();
            haveStudyCource++;

            //既然我已经学习过这个课程了,再去"谁依赖我的集合里面去  消消乐!!!"
            for(int willCourse:adjs[canStudyCource]){
                indegress[willCourse]--;
                if(indegress[willCourse]==0){
                    canStudy.add(willCourse);
                }
            }
        }

        return haveStudyCource==numCourses;
    }
}

题目链接

207. 课程表

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

C_x_330

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值