LeetCode HOT 100 —— 207 .课程表

news2025/1/16 8:12:01

题目

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

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

例如,先修课程对 [0, 1] 表示:想要学习课程 0 ,你需要先完成课程 1 。 请你判断是否可能完成所有课程的学习?如果可以,返回 true ;否则,返回 false 。

在这里插入图片描述

思路

拓扑排序:

用有向图表示依赖关系:

  • 示例:n = 6,先决条件表:[[3, 0], [3, 1], [4, 1], [4, 2], [5, 3], [5, 4]]
  • 0, 1, 2 没有先修课,可以直接选。其余的课,都有两门先修课
  • 用有向图来表示依赖关系:
    在这里插入图片描述
  • 上图是一个有向无环图,把一个 有向无环图 转成 线性的排序 就是 拓扑排序
  • 有向图中有 入度出度 两个概念,在本例中,顶点 0、1、2 的入度为 0,顶点 3、4、5 的入度为 2

选课顺序:

每次只能选入度为0的课,因为它不依赖别的课,如选了0,课3的入度由 2 变 1,接着选1,课 3 的入度变 0,课 4 的入度由 2 变 1,接着选 2,课 4 的入度变 0

然后现在,课 3 和课 4 的入度都为 0,继续选入度为 0 的课……直到选不到入度为 0 的课

类似BFS的思想:

  • 让入度为 0 的课入队列,它们是能直接选的课
  • 然后逐个出队列,出队列代表着该课被选,需要减小相关课(依赖出队列课的课)的入度
  • 如果相关课的入度新变为 0,安排它入队列、再出队列……直到没有入度为 0 的课可入队列

所需的数据结构:

  • 入度数组:课程号0到n-1作为索引,通过遍历先决条件表求出对应的初始入度
  • 邻接表:用哈希表记录依赖关系
    (1)key:课号
    (2)value: 依赖这门课的后续课(数组)

判断是否修完所有课程:

  1. BFS结束时,如果仍有课的入度不为0,无法被选,就无法完成所有课程
  2. 或者用一个变量count记录入队列的顶点个数,判断最后count是否等于总课程数

java代码如下:

class Solution{
	// 节点的入度: 使用数组保存每个节点的入度
	public boolean canFinish(int numCourses, int[][] prerequisites){
		// 1.课号和对应的入度
		Map<Integer,Integer> inDegree = new HashMap<>();
		// 将所有的课程先放入
		for(int i = 0; i < numCourses; i++){
			inDegree.put(i,0);
		}
		// 2.依赖关系, 依赖当前课程的后序课程
		Map<Integer,List<Integer>> adj = new HashMap<>();
		
		//初始化入度和依赖关系
		for(int[] relate : prerequisites){//prerequisites里面是一个数组列表,每个元素就是一个int[]数组
			// (3,0), 想学3号课程要先完成0号课程, 更新3号课程的入度和0号课程的依赖(邻接表),即next依赖于cur
			int cur = relate[1];//需要先学习的前驱课程,相当于(3,0)的0
			int next = relate[0];//完成前驱课程之后才能学习的课程,相当于(3,0)的3
			// 1.更新入度,因为next依赖于cur,所以增加入度
			inDegree.put(next,inDegree.get(next) + 1);
			// 2.当前节点的邻接表
			if(!adj.containsKey(cur)){//如果不包括前驱课程
				adj.put(cur, new ArrayList<>());//更新依赖图
			}
			adj.get(cur).add(next);//因为next依赖于cur,所以添加依赖关系
		}
		
		//3.BFS, 将入度为0的课程放入队列, 队列中的课程就是可以直接学的课程
		Queue<Integer> q = new LinkedList<>();
		for(int key : inDegree.keySet()){
			if(inDegree.get(key) == 0){
				q.offer(key);
			}
		}
		// 取出一个节点, 对应学习这门课程
		// 遍历当前邻接表, 更新其入度; 更新之后查看入度, 如果为0, 加入到队列
		while(!q.isEmpty()){
			int cur = q.poll();
			// 遍历当前课程的邻接表, 更新后继节点的入度
			if(!adj.containsKey(cur)){
				continue;
			}
			List<Integer> successList = adj.get(cur);//返回的是依赖于cur的数组列表
			
			for(int k : successList){//遍历这个列表,里面所有的入度减少一
				inDegree.put(k, inDegree.get(k) - 1);
				if(inDegree.get(k) == 0){//如果发现有入度为0的,则入队列
					q.offer(k);
				}
			}
		}
		
		// 4.遍历入队, 如果还有课程的入度不为0, 返回fasle
		for(int key : inDegree.keySet()){
			if(inDegree.get(key) != 0){
				return false;
			}
		}
		return true;
	}
}

总结:

  1. 根据依赖关系,建立邻接表、入度数组
  2. 选取入度为0的数据,根据邻接表,减小依赖它的数据的入度
  3. 找出入度为0的数据,重复第2步
  4. 直到所有数据的入度为0,得到排序,如果有数据入度不为0,说明图中存在环

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/78937.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

软件测试的类型

在本节中&#xff0c;我们将了解各种类型的软件测试&#xff0c;这些测试可以在软件开发生命周期时使用。 众所周知&#xff0c; 软件测试 是根据客户的先决条件分析应用程序功能的过程。 如果我们想确保我们的软件没有错误或稳定&#xff0c;我们必须执行各种类型的软件测试…

神经辐射场NeRF

NeRF: Representing Scenes as Neural Radiance Fields for Novel View Synthesis 文章目录NeRF: Representing Scenes as Neural Radiance Fields for Novel View SynthesisPipelineStepsVolume RenderingOptimizationPositional EncodingHierarchical Volume Samplingview-de…

从零学习gitHub (一)

一、账户注册 1.1、打开 GitHub 官方网址 GitHub 1.2、填入用户名「Username」、邮箱「Email」l、密码「Password 注意事项 Username&#xff1a;用户名不能重名&#xff0c;可包含字母数字字符和单行连字符&#xff0c;且不能以单行连字符开头或结尾&#xff1b; Email&…

Linux | 线程同步 | 条件变量 | 生产消费模型 | 阻塞队列实现生产消费模型

文章目录线程饥饿条件变量接口的使用生产者和消费者模型使用阻塞队列实现生产消费模型代码中存在的问题关于pthread_cond_wait的原子性生产消费模型中的并发体现线程饥饿 在多线程并发执行的场景中&#xff0c;会不会出现这样的情况&#xff0c;一些线程由于优先级更高&#x…

分布式锁4-Redisson分布式锁实现与看门狗原理

文章目录一.Redisson介绍二.分布式锁的运用1.引入依赖.2.增加配置类.3.简单代码实现1.不指定加锁时间,会默认启动看门狗.自动帮你的锁进行续期.2.指定加锁时间,不启用看门狗续期,到期自动解锁.三.分布式锁实现原理加锁过程看门狗续期过程一.Redisson介绍 Redisson 是架设在 Red…

读《冯诺依曼传》

关于冯诺依曼 冯诺依曼是20世纪的全才,原名约翰尼,匈牙利美籍科学家,被称为计算机之父和博弈论之父。计算机和博弈论都深刻改变人类的生活工作和思维方式,极大地促进了社会和人类文明的进步发展。在六个哲学领域他都做出了相当大的贡献,把模糊的问题用数学精确地表述出来…

扩散模型的迁移和应用(DiffusionDet,AR-LDM,MagicVideo,RenderDiffusion,AltDiffusion,VD)

在上一篇博文中&#xff0c;博主已经整理了扩散模型&#xff08;Diffusion Model&#xff0c;DDPM&#xff0c;GLIDE&#xff0c;DALLE2&#xff0c;Stable Diffusion&#xff09;的基本原理&#xff0c;后续不再赘述其细节。作为一个最近被讨论热烈的方向&#xff0c;很自然地…

干Java开发快30岁了,没有核心竞争力,该怎么跳槽面试?

今年互联网大环境的确不怎么好&#xff0c;互联网公司纷纷陷入裁员风波并缩减规模&#xff0c;这就导致更多程序员需要去竞争更少的就业岗位&#xff0c;搞的整个IT 行业越来越卷。作为Java程序员的我们就更不用说了&#xff0c;除去加班的可能性&#xff0c;每天上班8小时需要…

Java项目:SSM药品进货销售仓储信息管理系统

作者主页&#xff1a;源码空间站2022 简介&#xff1a;Java领域优质创作者、Java项目、学习资料、技术互助 文末获取源码 项目介绍 管理员角色包含以下功能&#xff1a; 管理员登录,用户信息管理,个人信息管理,药品信息管理,药品类别信息管理,选购药品管理,订单管理,订单记录管…

余生很短,及时止损--爱摸鱼的美工(九)

-----------作者&#xff1a;天涯小Y 余生很短&#xff0c;及时止损 今日份垃圾清理完毕 有的人就像垃圾车 他们装满了垃圾四处奔走 充满懊悔、愤怒.失望的情绪 随着垃圾越堆越高 就需要找地方倾倒&#xff0c;释放出来 他们会逮着一切机会 到处碰瓷、找茬、泄愤 如果你给他们…

Js逆向教程22-AST 抽象语法树babel安装

作者&#xff1a;虚坏叔叔 博客&#xff1a;https://xuhss.com 早餐店不会开到晚上&#xff0c;想吃的人早就来了&#xff01;&#x1f604; AST 抽象语法树babel安装 一、什么是AST js代码&#xff1a; var a1;json标识&#xff1a; 赋值语句左边为定义定义的名称为a(作用域…

python——面向对象

1、概念 ①OOP&#xff1a;Object Oriented Programming ②面向对象&#xff1a;只关心哪个对象完成这个功能&#xff0c;它关注的是完成功能的结果&#xff0c;以结果为导向 python、Java、js、c... ③面向过程&面向对象 ④面向对象三大特征&#xff1a;封装、继承、多…

springcloud(配置中心)

目录 1. 为何需要配置中心2. nacos的配置中心 2.1 配置示例2.2 多环境配置2.3 GROUP2.4 namespace 1. 为何需要配置中心 没有配置中心的情况下会出现什么情况&#xff1a; 如果没有配置中心&#xff0c;则各个服务的配置文件分散在各个项目中&#xff0c;不方便维护出于安全考…

代码审计学习 : xhcms

前言 从大佬那里听说&#xff0c;xhcms 很适合代码审计的新手。 环境 php 5.4.45 Apache 2.4.39 Mysql 5.7.26 文件上传配合文件包含 /index.php 和 /admin/index.php error_reporting(0); //关闭错误显示 $fileaddslashes($_GET[r]); //接收文件名 $action$file?index:…

FH30502输入3.7V升5V电流3A-5A同步整流升压芯片

3.3V升5V电流3A-5A同步整流升压芯片&#xff0c;2.7V到18V的输入电压支持供电系统和电池的较宽范围应用。FH30502根据负载情况的变化自动切换工作模式&#xff0c;在轻载Burst模式下静态电流处于低状态。FH30502使用自适应常数断开时间峰值电流模式控制。FH30502有一个内部特性…

C 语言实现经典贪吃蛇游戏

原文链接&#xff1a;C语言 贪吃蛇游戏 文章目录一、说明二、效果2.1 欢迎界面2.2 游戏规则2.3 得分排行2.4 退出游戏2.5 游戏界面2.6 游戏结束三、源码3.1 cmd.h3.2 cmd.c3.3 io.h3.4 io.c3.5 model.h3.6 service.h3.7 service.c3.8 ui.h3.9 ui.c3.10 utils.h3.11 utils.c3.12…

m蜂窝移动通信系统中越区切换的matlab仿真分析

目录 1.算法描述 2.仿真效果预览 3.MATLAB核心程序 4.完整MATLAB 1.算法描述 用Matlab仿真一个蜂窝模型&#xff0c;用户在打电话时产生切换的过程。建立蜂窝越区切换&#xff0c;假设有16个蜂窝&#xff0c;实现两个小区之间的硬切换&#xff0c;每个小区假设能容纳30个用…

UDS入门至精通系列:Service 14

文章目录 一、Service 14功能是什么二、UDS协议对服务的定义三、用图形说明Service 14四、手动测试总结一、Service 14功能是什么? 在汽车电子诊断领域,在新车型定义诊断需求时,会给每一个ECU的故障类型定义一个DTC,ECU中运行代码判定DTC是否产生(判定机制和原理我会在关于…

Gateway之限流、熔断

目录 一、Sentinel--服务容错 ① 简化启动我们的nacos 1.高并发带来的问题 ① 修改配置文件中tomcat的并发数 ② 使用压测工具&#xff0c;对请求进行压力测试 第一步&#xff1a;修改配置&#xff0c;并启动软件 第二步&#xff1a;添加线程组 第三步&#xff1a;配置线…

MySQL 小版本升级步骤

MySQL mysql 5.7.38 升级到 5.7.40 下载软件升级包 参考下载地址&#xff1a;https://dev.mysql.com/get/Downloads/MySQL-5.7/mysql-5.7.40-linux-glibc2.12-x86_64.tar.gz 上传软件包至服务器 可以使用rz命令或scp上传本地文件 比如上传至服务器目录为 /usr/local/software…