【算法】【优选算法】BFS 解决拓扑排序

【算法】【优选算法】BFS 解决拓扑排序

目录

一、拓扑排序

1.1 有向无环图(DAG图)

有向无环图:有向无环图:一个无回路的有向图,如果一个有向图无法从某个顶点出发经过若干条边回到该点,则这个图是一个有向无环图(DAG图)。

1.2 AOV 网:顶点活动图

在有向无环图的基础上,用顶点来表示一个活动,用边来表示活动执行的先后顺序。

1.3 拓扑排序

拓扑排序: 拓扑排序,找到做事的先后顺序。每次出入度为0的点,并且删除该点相连的边,排序进行。

1.4 实现拓扑排序

  1. 找出图中入度为0的点
  2. 删除与该点连接的变
  3. 重复1 2 直到没有点 或者 没有入度为0 的点。

借助队列,来一次BFS

  1. 初始化:把所有入度为0的点加入到队列
  2. 当队列不为空:
    1. 拿出队头元素,加入到最终结果
    1. 删除与元素相连的边
    1. 判断:与删除的边相连的点,入度是否变为0

二、207. 课程表

题目链接:207. 课程表

题目描述:

题目解析:

  • 给我们一个int数,代表id为 0 到 numCourses 的课程,
  • 给我们一个 prerequisites 数组,其中从后往前代表学习顺序
  • 让我们看能否排课,将课程学完,必须保证学习顺序

解题思路:

  • 我们使用二维数组List<List>,来表示图,每一个下标对应的一维数组,表示该点指向的点;
  • 我们遍历一次prerequisites数组,将最后的n-1列的值当成点,其余入该点对应的一维数组。除了n-1列,其余对应的入度数组对应下标值加一。
  • 在进行一次BFS即可
  • 注意事项:每个点只能入一次队,prerequisites可以是 [ ] 数组,在前面要排除。
    解题代码:
//O(M*N)//O(M*N)classSolution{publicbooleancanFinish(int numCourses,int[][] prerequisites){int m = prerequisites.length;if(m ==0)returntrue;int n = prerequisites[0].length;//建图List<List<Integer>> edges =newArrayList<>();for(int i =0; i < numCourses; i++){ edges.add(newArrayList<>());}Queue<Integer> queue =newLinkedList<>();//记录下标对应每个点的入度int[] in =newint[numCourses];//标记入过队的元素boolean[] flag =newboolean[numCourses];//记录入度for(int i =0; i < m; i++){for(int j =0; j < n-1; j++){//建图 edges.get(prerequisites[i][n-1]).add(prerequisites[i][j]);//统计入度 in[prerequisites[i][j]]++;}}//入度为0的点入队for(int i =0; i < numCourses; i++){if(in[i]==0){ queue.add(i); flag[i]=true;}}if(queue.isEmpty())returnfalse;//BFSwhile(!queue.isEmpty()){int tmp = queue.poll();//销毁边for(int i =0; i < edges.get(tmp).size(); i++){ in[edges.get(tmp).get(i)]--;}//入度为0的点入队for(int i =0; i < numCourses; i++){if(in[i]==0&&!flag[i]){ queue.add(i); flag[i]=true;}}}//还有入度不为0的点for(int i =0; i < numCourses; i++){if(in[i]!=0){returnfalse;}}returntrue;}}

三、210. 课程表 II

题目链接:210. 课程表 II

题目描述:

题目解析:

  • 给我们一个int数,代表id为 0 到 numCourses 的课程,
  • 给我们一个 prerequisites 数组,其中从后往前代表学习顺序
  • 让我们看能否排课,将课程学完,必须保证学习顺序,返回学习的顺序,不行就返回空数组。
    解题思路:
  • 我们使用二维数组List<List>,来表示图,每一个下标对应的一维数组,表示该点指向的点;
  • 我们遍历一次prerequisites数组,将最后的n-1列的值当成点,其余入该点对应的一维数组。除了n-1列,其余对应的入度数组对应下标值加一。
  • 在进行一次BFS即可,每次出的点都进入结果数组
  • 注意事项:
    • 我们先将结果数组初始化为 0 到 numCourses-1 ,,prerequisites可以是 [ ] 数组时直接就返回
    • 当最后执行完,我们使用一个计数器lep来标记进入结果数组的元素,最后只需要比较lep与numCourses是否一样,一样返回结果数组,不一样返回空数组。
      解题代码:
//O(M*N)//O(M*N)classSolution{publicint[]findOrder(int numCourses,int[][] prerequisites){//结果数组int[] ret =newint[numCourses];for(int i =0; i < numCourses;i++) ret[i]= i;int m = prerequisites.length;if(m ==0)return ret;int n = prerequisites[0].length;Queue<Integer> queue =newLinkedList<>();//标记数组boolean[] flag =newboolean[numCourses];//入度数组int[] in =newint[numCourses];//建图List<List<Integer>> edges =newArrayList<>();for(int i =0; i < numCourses; i++){ edges.add(newArrayList<>());}//记录入度,初始图for(int i =0; i < m; i++){for(int j =0; j < n-1; j++){ edges.get(prerequisites[i][n-1]).add(prerequisites[i][j]); in[prerequisites[i][j]]++;}}//入度为0的点入队for(int i =0; i < numCourses; i++){if(in[i]==0){ flag[i]=true; queue.add(i);}}//BFSint lep =0;while(!queue.isEmpty()){int tmp = queue.poll(); ret[lep++]= tmp;//删去该点出的边for(int i =0; i < edges.get(tmp).size(); i++){ in[edges.get(tmp).get(i)]--;}//入队for(int i =0; i < numCourses; i++){if(in[i]==0&&!flag[i]){ queue.add(i); flag[i]=true;}}}if(lep != numCourses)returnnewint[]{};return ret;}}

四、LCR 114. ⽕星词典

题目链接:LCR 114. ⽕星词典

题目描述:

题目解析:

  • 给我们一个字符串数组words,
  • 两个字符串对比来得出字母顺序,两个字符串第一个不相同的字母的顺序 等于 两个字符串在数组中的顺序,如示例一第一个字符串“wrt” 和 第二个字符串“wrf” 得出的字母顺序就是 t->f
  • 将所有的字符两两对比,得出字母顺序,不能得出线性顺序就返回空字符串“”

解题思路:

  • 两层for循环,先搜索两个规则导致的排序结果建立图,例如示例一就建立这个图
  • 根据图拓扑排序 ,得出结论
  • 细节问题: abc 和 ab,并且abc在数组中位于ab前面,这种是不合法的
    解题代码:
//O(M*N)//O(M*N)classSolution{//图Map<Character,Set<Character>> edges =newHashMap<>();//入度Map<Character,Integer> in =newHashMap<>();//标记有没有abc ab这种不合法boolean flag;publicStringalienOrder(String[] words){//初始化入度for(String s : words){for(int i =0; i < s.length(); i++){char ch = s.charAt(i); in.put(ch,0);}}//建图for(int i =0; i < words.length; i++){for(int j = i +1; j <words.length; j++){//统计两个字符串的字母顺序add(words[i], words[j]);if(flag ==true)return"";}}//拓扑排序初始化Queue<Character> queue =newLinkedList<>();for(char ch : in.keySet()){if(in.get(ch)==0) queue.add(ch);}//BFSStringBuffer ret =newStringBuffer();while(!queue.isEmpty()){char tmp = queue.poll(); ret.append(tmp);if(!edges.containsKey(tmp))continue;for(char ch : edges.get(tmp)){ in.put(ch, in.get(ch)-1);if(in.get(ch)==0){ queue.add(ch);}}}for(char ch : in.keySet()){if(in.get(ch)!=0)return"";}return ret.toString();}publicvoidadd(String a,String b){int i =0;int n =Math.min(a.length(), b.length());for(; i < n; i++){char ch1 = a.charAt(i);char ch2 = b.charAt(i);if(ch1 != ch2){//ch1 -> ch2if(!edges.containsKey(ch1)){ edges.put(ch1,newHashSet<>());}if(!edges.get(ch1).contains(ch2)){ edges.get(ch1).add(ch2); in.put(ch2,in.get(ch2)+1);}break;}}if(i == b.length()&& a.length()> i) flag =true;}}

Read more

Flutter 三方库 xpath_selector 的鸿蒙化适配指南 - 在鸿蒙系统上构建极致、透明、精准的 HTML/XML 数据抓取与 Web 结构解析引擎

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 三方库 xpath_selector 的鸿蒙化适配指南 - 在鸿蒙系统上构建极致、透明、精准的 HTML/XML 数据抓取与 Web 结构解析引擎 在鸿蒙(OpenHarmony)系统的网络爬虫、自动化测试审计、或者是从复杂的第三方 Web 公告(HTML)中提取关键数据(如新闻标题、资产负债表)时,如何摆脱凌乱的正向正则(Regex),转而使用业界标准的 XPath 语法进行语义化选取?xpath_selector 为开发者提供了一套工业级的、基于 Dart 的 HTML/XML 结构化查询方案。本文将深入实战其在鸿蒙端数据治理中的应用。 前言 什么是 XPath Selector?

By Ne0inhk

Flutter 三方库 bones_ui 的鸿蒙化适配指南 - 打造直观、响应式的 Web 风格 UI 交互体验

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 三方库 bones_ui 的鸿蒙化适配指南 - 打造直观、响应式的 Web 风格 UI 交互体验 Flutter for OpenHarmony 开发者在构建具有 Web 质感的跨平台应用时,UI 框架的选择至关重要。本文将带大家深度调研 Dart 三方库 bones_ui 在鸿蒙系统上的适配方案,探索如何利用其直观的组件架构,加速鸿蒙桌面级应用的开发效率。 前言 在移动端和桌面端融合的今天,开发者往往希望一套代码能同时适配多种屏幕形态。bones_ui 原生为 Dart Web 打造,但在 Flutter for OpenHarmony 的大前端生态中,其简洁的 UI 组件设计思想对我们构建鸿蒙跨平台应用具有极大的参考价值。

By Ne0inhk
【数据结构-初阶】顺序表相关习题

【数据结构-初阶】顺序表相关习题

🎈主页传送门:良木生香 🔥个人专栏:《C语言》 《数据结构-初阶》 《程序设计》 🌟人为善,福随未至,祸已远行;人为恶,祸虽未至,福已远离 上期回顾:在上一篇文章中(【数据结构-初阶】详解线性表(1)---顺序表),我们详细介绍了线性表系列第一种数据结构---顺序表,这个数据结构是以数组为底建立的,也学习了如何用线性表进行增删查改的操作,那么我们今天就用顺序表进行解题~~~   题目一:移除元素 这是题目链接:27.移除元素,下面是具体的题目与示例: 由题意知,这道题是想让我们将数组中值为val的元素删除,我们能怎么做呢? 创建新的数组?那不行,题目已经要求我们只能在原地进行操作了,就意味着不能创建新的数组来进行辅助 那该怎么办呢?简单,我们只需用上算法中最基础的---双指针算法了 我们用双指针,不一定用真的指针指向某个元素,有时也可以用下标,讲究的是一种算法思想,并没有一定的形式 我们用两个指针,刚开始都同事之下那个num数组的第一个元素,随后将其中一个指针用于遍历数组,如果两个指针指向的内容不相同,那就将内容进行交换,两个指针同时向后移动一位;如果相同

By Ne0inhk

黑马程序员java web学习笔记--后端进阶(二)SpringBoot原理

目录 1 配置优先级 2 Bean的管理 2.1 Bean的作用域 2.2 第三方Bean 3 SpringBoot原理 3.1 起步依赖 3.2 自动配置 3.2.1 实现方案 3.2.2 原理分析 3.2.3 自定义starter 1 配置优先级 SpringBoot项目当中支持的三类配置文件: * application.properties * application.yml ❤ * application.yaml 配置文件优先级排名(从高到低):properties配置文件 > yml配置文件 > yaml配置文件 虽然springboot支持多种格式配置文件,但是在项目开发时,推荐统一使用一种格式的配置。

By Ne0inhk