【算法】【优选算法】BFS 解决拓扑排序
目录
一、拓扑排序
1.1 有向无环图(DAG图)
有向无环图:有向无环图:一个无回路的有向图,如果一个有向图无法从某个顶点出发经过若干条边回到该点,则这个图是一个有向无环图(DAG图)。

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

1.3 拓扑排序
拓扑排序: 拓扑排序,找到做事的先后顺序。每次出入度为0的点,并且删除该点相连的边,排序进行。
1.4 实现拓扑排序
- 找出图中入度为0的点
- 删除与该点连接的变
- 重复1 2 直到没有点 或者 没有入度为0 的点。
借助队列,来一次BFS
- 初始化:把所有入度为0的点加入到队列
- 当队列不为空:
- 拿出队头元素,加入到最终结果
- 删除与元素相连的边
- 判断:与删除的边相连的点,入度是否变为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;}}