数据结构与算法:图论——拓扑排序

基础与模板:

有两个KahnDFS两个算法

下面给出Kahn的算法模板

c 复制代码
#include<iostream>
#include<vector>
#include<queue>
using namespace std;

vector<int> topologicalSortKahn(int num, const vector<pair<int, int>>& relations){
	vector<int> result;
	vector<int> inDegree(num+1,0);	// 记录所有节点的入度
	vector<vector<int>> adj(num+1);	// 用于记录节点的连接的节点
    
	for(const auto &rel:relations){
		int from = rel.first;
		int to	 = rel.second;
		adj[from].push_back(to);	// 写入每个节点连接的节点
		inDegree[to]++;				// 写入入度
	}
	
	queue<int> q;					// 用于记录入度为0的节点
	// 遍历节点,把入度为0的点写入q
	// 这里需要注意节点到底是从0开始还是从1开始
	for(int i = 0;i< num;++i){
		if(inDegree[i] == 0){
			q.push(i);
		}
	}
	// 提取入度为0的节点,删除其连接,不断更新入度
	while(!q.empty()){
		int current= q.front();
		q.pop();
		result.push_back(current);		
		for(int next : adj[current]){
			inDegree[next]--;		// 每个下一节点的入度减1
			if(inDegree[next]==0){	// 入度减1后判断为0加入q
				q.push(next);
			}
		}
	}
    if(result.size() != num) return{-1}; // 判断是环,返回-1
	return result;
}

int main(){
	int N,M;
	cin>>N>>M; 
	vector<pair<int,int>>  inRelation;
	int from,to;
	while(M--){
		cin>>from>>to;
		inRelation.emplace_back(pair<int,int>{from,to});
	}
	vector<int> res = topologicalSortKahn(N,inRelation);
	for(size_t i = 0;i < res.size()-1;i++){
		cout<<res[i]<<" ";
	}
	cout<<res.back();
}

图的拓扑排序题目leetcode

题号 标题 题解 标签 难度
0207 课程表 Python 深度优先搜索、广度优先搜索、图、拓扑排序 中等
0210 课程表 II Python 深度优先搜索、广度优先搜索、图、拓扑排序 中等
1136 并行课程 Python 图、拓扑排序 中等
2050 并行课程 III Python 图、拓扑排序、数组、动态规划 困难
0802 找到最终的安全状态 Python 深度优先搜索、广度优先搜索、图、拓扑排序 中等
0851 喧闹和富有 Python 深度优先搜索、图、拓扑排序、数组 中等

题目1、117. 软件构建

卡吗网代码编辑真不错

直接套模板

题目2、210. 课程表 II

c 复制代码
class Solution {
public:
	vector<int> findOrder(int numCourses, vector<vector<int>>& prerequisites) {
		vector<int> result;
		vector<int> inDegree(numCourses+1,0);	// 记录所有节点的入度
		vector<vector<int>> adj(numCourses+1);	// 用于记录节点的连接的节点
		for(const auto &rel:prerequisites){
			int from = rel[0];
			int to	 = rel[1];
			adj[from].push_back(to);	// 写入每个节点连接的节点
			inDegree[to]++;				// 写入入度
		}
		
		queue<int> q;					// 用于记录入度为0的节点
		// 遍历节点,把入度为0的点写入q
		// 这里需要注意节点到底是从0开始还是从1开始
		for(int i = 0;i< numCourses;++i){
			if(inDegree[i] == 0){
				q.push(i);
			}
		}
		// 提取入度为0的节点,删除其连接,不断更新入度
		while(!q.empty()){
			int current= q.front();
			q.pop();
			result.push_back(current);		
			for(int next : adj[current]){
				inDegree[next]--;		// 每个下一节点的入度减1
				if(inDegree[next]==0){	// 入度减1后判断为0加入q
					q.push(next);
				}
			}
		}
		if(result.size() != static_cast<size_t>(numCourses)) return{}; // 判断是环,返回{}
		return result;
	}
};

题目3、任务调度算法



c 复制代码
#include <vector>
#include <utility>
#include <queue>
#include <algorithm>


using namespace std;

class Solution {
public:
	int GetMinT	ime(int taskNum, const vector<pair<int, int>>& relations)
	{
		vector<vector<int>> adj(taskNum+1);
		vector<int> inDegree(taskNum + 1, 0);
		vector<int> dp(taskNum + 1, 1);
		
		for (const auto& rel : relations) {
			int from = rel.second; 
			int to = rel.first;    
			adj[from].push_back(to);
			inDegree[to]++;
		}
		queue<int> q;
		for (int i = 1; i <= taskNum; ++i) {
			if (inDegree[i] == 0) {
				q.push(i);
			}
		}
		
		while (!q.empty()) {
			int current = q.front();
			q.pop();
			for (int next : adj[current]) {
				dp[next] = max(dp[next], dp[current] + 1);
				if (--inDegree[next] == 0) {
					q.push(next);
				}
			}
		}
		return *max_element(dp.begin() + 1, dp.end());

	}
};

题目4、207. 课程表 - 力扣

这里就是判断是否有环

相关推荐
辰海Coding5 小时前
MiniSpring框架学习笔记-解决循环依赖的简化IoC容器
笔记·学习
晓梦林6 小时前
cp520靶场学习笔记
android·笔记·学习
小鹏linux6 小时前
Ubuntu 22.04 部署开源免费具有精美现代web页面的Casdoor账号管理系统
linux·前端·ubuntu·开源·堡垒机
在角落发呆7 小时前
Linux转发配置:解锁网络互联的核心密码
linux·运维·网络
心中有国也有家7 小时前
cann-recipes-infer:昇腾 NPU 推理的“菜谱集合”
经验分享·笔记·学习·算法
Upsy-Daisy7 小时前
AI Agent 项目学习笔记(八):Tool Calling 工具调用机制总览
人工智能·笔记·学习
齐潇宇7 小时前
Zabbix 7 概述与配置
linux·zabbix·监控告警
LuminousCPP8 小时前
数据结构 - 线性表第四篇:C 语言通讯录优化升级全记录(踩坑 + 思考)
c语言·开发语言·数据结构·经验分享·笔记·学习
江公望8 小时前
Ubuntu htop命令,10分钟讲清楚
linux·服务器
哎呦,帅小伙哦8 小时前
Linux 时间:从原子钟到 clock_gettime 的每一面
linux·运维·服务器