207. 课程表

文章目录

题目

图论:207. 课程表

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

在选修某些课程之前需要一些先修课程。 先修课程按数组 prerequisites 给出,其中 prerequisitesi = 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

prerequisitesi.length == 2

0 <= ai, bi < numCourses

prerequisitesi 中的所有课程对 互不相同

代码

cpp 复制代码
class Solution {
public:
    bool canFinish(int numCourses, vector<vector<int>>& prerequisites) {
      vector<int>ingree(numCourses,0);//入度数组:记录每门课有多少先修课还没修
      unordered_map<int,vector<int>>preCourse;//邻接表:key=课程,value=学完这门课后可以解锁的后续课程
      for(int i=0;i<prerequisites.size();i++){
        // 前提:prerequisites[i][0] 想要上课,必须先上 prerequisites[i][1]
        // 所以建边:prerequisites[i][1] → prerequisites[i][0]
        preCourse[prerequisites[i][1]].push_back(prerequisites[i][0]);
        // 0号课程的入度 +1(因为它依赖1号课程)
        ingree[prerequisites[i][0]]++;
      }
      // BFS队列:存放入度为0(没有先修课,可以直接上)的课程
      queue<int>q;
    // 统计已经上完的课程数
      int count=0;
      //初始化队列:把所有入度为0的课放进去
      for(int i=0;i<numCourses;++i){
        if(!ingree[i]){
            q.push(i);
            ++count;//这门课可以直接上,计数+1
        }
      }
      while(!q.empty()){//队列不为空的前提下
        int cur=q.front();//取出队头
        q.pop();//删除队头
        //遍历这门课能解锁的所有后续课程
        for(int i=0;i<preCourse[cur].size();++i){
            --ingree[preCourse[cur][i]];//队头依赖的课程的入度减1
            // 如果入度变成0 → 没有先修课了,可以上了
            if(!ingree[preCourse[cur][i]]){//将度为0的课程压入队列
                q.push(preCourse[cur][i]);
                ++count;
            }
        }
      }
    if(count==numCourses)return true;
    return false;
    }
};

原理图

原理解释

提示:算法流程及解释在代码中已标注

判断有向图中是否存在环,如果存在环说明互相依赖为false

如果不存在环说明可以是实现学完所有课程为true

拓扑排序 + 广度优先搜索 (BFS)

1、初始化一个入度数组,记录每门课有多少先修课还没完成。

2、建立邻接表,表示课程之间的依赖关系:学完 A 可以解锁哪些课。

3、遍历所有先修条件,把依赖关系填入邻接表,并更新对应课程的入度。

4、遍历所有课程,把入度为 0(没有先修课、可以直接学)的课程加入队列。

5、用计数器记录已经修完的课程数量,初始为入度 0 的课程数。

6、开始 BFS 循环:取出队列头部的一门课,表示当前修完这门课。

7、遍历这门课能解锁的所有后续课程,将它们的入度减 1。

8、若某后续课程入度减到 0,说明它的先修课都修完了,加入队列并增加计数。

9、重复直到队列为空。

10、最后判断:修完的课程总数是否等于总课程数。

相等 → 无环,可以修完所有课 → 返回 true

不相等 → 有环,无法修完 → 返回 false

相关推荐
upgrador2 分钟前
基础知识:C++ STL构造函数的左闭右开惯例及其实现原理
开发语言·c++
我是一颗柠檬1 小时前
【Java项目技术亮点】加权轮询负载均衡算法
java·算法·负载均衡
灯厂码农1 小时前
C语言动态内存分配完全指南(malloc、calloc、realloc、free)
java·c语言·算法
凯瑟琳.奥古斯特2 小时前
K次取反最大化数组和解法(力扣1005)
开发语言·c++·算法·leetcode·职场和发展
林中青木2 小时前
CT重构原理及C++代码实现
c++·计算机视觉·重构
满天星83035772 小时前
Protobuf的介绍及使用
c++
☆cwlulu3 小时前
调试排查工具介绍(gdb、strace、Valgrind等)
开发语言·c++·嵌入式硬件·ubuntu
Jerry3 小时前
LeetCode 203. 移除链表元素
算法
地平线开发者3 小时前
征程 6 | 工具链 QAT ObserverBase 源码解析
算法