当前位置: 首页 > article >正文

力扣面试150题--课程表

Day 63

题目描述

在这里插入图片描述

做法

初次思路:本质就是将所有前置课程和后置课程作为一个有向图(前者指向后者),判断这个图是否是一个有向无环图(即是否存在拓扑排序)(本质做法是dfs)
做法

  1. 将所有的前置和后置课程初始化为一个邻接矩阵graph
  2. 在初始化的过程中找到一个入度为0的课程作为起点进入find函数(这里我的做法比较浪费,直接使用了一个hash数组来找)
  3. 进入find函数来找拓扑排序
  4. visited判断该点是否访问过,如果访问过,则find函数直接结束,如果没有访问过,将其visited置为1
  5. 将这个课程压入栈中
  6. 判断所有将这个课程作为前置条件的课程,是否可以作为下一个课程(check函数的作用就是判断将这个课程作为前置课程的课程是否还有其他的前置课程)
  7. 如果可以,就接着对这个课程执行find函数
  8. 最后栈中的所有值就是拓扑排序的倒序
  9. 如果栈中元素等于所有课程数,就可以完全选择,不等于就不行。
class Solution {public Stack<Integer>stack=new Stack<Integer>();public boolean check(int[][]graph,int x,int[]visited){boolean z=true;for(int i=0;i<graph.length;i++){if(graph[i][x]==1&&visited[i]==0){//如果这个点还需要其他入度说明这个点不能选z=false;break;}}return z;}public void find(int[][]graph,int[] visited,int i){if(visited[i]==1){return ;}visited[i]=1;stack.push(i);for(int x=0;x<graph.length;x++){if(graph[i][x]==1&&check(graph,x,visited)){//说明相连find(graph,visited,x);}}}public boolean canFinish(int numCourses, int[][] prerequisites) {int[][] graph=new int[numCourses][numCourses];int[] visited=new int[numCourses];int [] beg=new int[numCourses];int x,y;for(int i=0;i<prerequisites.length;i++){x=prerequisites[i][0];//后置y=prerequisites[i][1];//前提beg[x]=1;//有入度的全部设置为1,方便后面找入度为0的点graph[y][x]=1;}//找个入度为0的起点for(int i=0;i<numCourses;i++){if(beg[i]==0){find(graph,visited,i);}}if(stack.size()==numCourses){return true;}return false;}
}

题解思路
我们使用一个队列来进行广度优先搜索。初始时,所有入度为 0 的节点都被放入队列中,它们就是可以作为拓扑排序最前面的节点,并且它们之间的相对顺序是无关紧要的。

在广度优先搜索的每一步中,我们取出队首的节点 u:

我们将 u 放入答案中;

我们移除 u 的所有出边,也就是将 u 的所有相邻节点的入度减少 1。如果某个相邻节点 v 的入度变为 0,那么我们就将 v 放入队列中。

在广度优先搜索的过程结束后。如果答案中包含了这 n 个节点,那么我们就找到了一种拓扑排序,否则说明图中存在环,也就不存在拓扑排序了。

class Solution {List<List<Integer>> edges;int[] visited;boolean valid = true;public boolean canFinish(int numCourses, int[][] prerequisites) {edges = new ArrayList<List<Integer>>();for (int i = 0; i < numCourses; ++i) {edges.add(new ArrayList<Integer>());}visited = new int[numCourses];for (int[] info : prerequisites) {edges.get(info[1]).add(info[0]);}for (int i = 0; i < numCourses && valid; ++i) {if (visited[i] == 0) {dfs(i);}}return valid;}public void dfs(int u) {visited[u] = 1;for (int v: edges.get(u)) {if (visited[v] == 0) {dfs(v);if (!valid) {return;}} else if (visited[v] == 1) {valid = false;return;}}visited[u] = 2;}
}
http://www.lryc.cn/news/2405047.html

相关文章:

  • 用通俗的话解释下MCP是个啥?
  • LeetCode 高频 SQL 50 题(基础版)之 【子查询】· 上
  • Spark流水线+Gravitino+Marquez数据血缘采集
  • 一个完整的时间序列异常检测系统,使用Flask作为后端框架,实现了AE(自编码器)、TimesNet和LSTM三种模型,并提供可视化展示
  • 深度学习在非线性场景中的核心应用领域及向量/张量数据处理案例,结合工业、金融等领域的实际落地场景分析
  • 基于微信小程序的车位共享平台的设计与实现源码数据库文档
  • 多模态大语言模型arxiv论文略读(111)
  • 网页端 VUE+C#/FastAPI获取客户端IP和hostname
  • 一个自动反汇编脚本
  • 函数与数列的交汇融合
  • 怎么让自己ip显示外省?一文说清操作
  • 【Docker】容器安全之非root用户运行
  • 汽车车载软件平台化项目规模颗粒度选择的一些探讨
  • 【八股消消乐】构建微服务架构体系—服务注册与发现
  • 大数据+智能零售:数字化变革下的“智慧新零售”密码
  • C++_核心编程_菱形继承
  • 掌握Git核心:版本控制、分支管理与远程操作
  • c#,Powershell,mmsys.cpl,使用Win32 API展示音频设备属性对话框
  • STM标准库-TIM旋转编码器
  • 深入解析JVM工作原理:从字节码到机器指令的全过程
  • MCP通信方式之Streamable HTTP
  • 第七十三篇 从电影院售票到停车场计数:生活场景解析Java原子类精髓
  • 【原创】基于视觉模型+FFmpeg+MoviePy实现短视频自动化二次编辑+多赛道
  • C++----剖析list
  • 纳米AI搜索与百度AI搜、豆包的核心差异解析
  • 不到 2 个月,OpenAI 火速用 Rust 重写 AI 编程工具。尤雨溪也觉得 Rust 香!
  • 人工智能:网络安全的“智能守护者”
  • Python60日基础学习打卡Day46
  • 综述论文解读:Editing Large Language Models: Problems, Methods, and Opportunities
  • WEB3全栈开发——面试专业技能点P1Node.js / Web3.js / Ethers.js