洛谷-P1807-最长路

最长路

题目描述

G G G 为有 n n n 个顶点的带权有向无环图, G G G 中各顶点的编号为 1 1 1 n n n,请设计算法,计算图 G G G 1 , n 1, n 1,n 间的最长路径。

输入格式

输入的第一行有两个整数,分别代表图的点数 n n n 和边数 m m m

2 2 2 到第 ( m + 1 ) (m + 1) (m+1) 行,每行 3 3 3 个整数 u , v , w u, v, w u,v,w u < v u<v u<v),代表存在一条从 u u u v v v 边权为 w w w 的边。

输出格式

输出一行一个整数,代表 1 1 1 n n n 的最长路。

1 1 1 无法到达 n n n,请输出 − 1 -1 1

样例 #1

样例输入 #1

2 1
1 2 1

样例输出 #1

1

提示

【数据规模与约定】

  • 对于 20 % 20\% 20%的数据, n ≤ 100 n \leq 100 n100 m ≤ 1 0 3 m \leq 10^3 m103
  • 对于 40 % 40\% 40% 的数据, n ≤ 1 0 3 n \leq 10^3 n103 m ≤ 1 0 4 m \leq 10^{4} m104
  • 对于 100 % 100\% 100% 的数据, 1 ≤ n ≤ 1500 1 \leq n \leq 1500 1n1500 0 ≤ m ≤ 5 × 1 0 4 0 \leq m \leq 5 \times 10^4 0m5×104 1 ≤ u , v ≤ n 1 \leq u, v \leq n 1u,vn − 1 0 5 ≤ w ≤ 1 0 5 -10^5 \leq w \leq 10^5 105w105

#2寄的是因为没有初始化距离为负无穷,会导致负权边取错最大距离

#include<iostream>
#include<cstring>
#include<vector>
#include<queue>
using namespace std;
const int N = 5 * 1e4 + 10;

struct Edge {
	int to;
	int w;
};
vector<Edge> e[N];	//二维结构体vector存图

int n, m;
int indegree[N];	//入度
int outdegree[N];	//出度
long long dis[N];	//距离数组

void topsort() {	//拓扑排序
	queue<int>q;
	q.push(1);
	dis[1] = 0;

	while (q.size()) {
		int t = q.front();
		q.pop();

		for (int i = 0; i < e[t].size(); i++) {
			int j = e[t][i].to, ww = e[t][i].w;
			indegree[j]--;

			dis[j] = max(dis[t] + ww, dis[j]);	//取最长的边
			//每次更新时都判断一下,是dis[j]现存的距离更大,还是dis[t]+w更大,找最大的一条边

			if (indegree[j] == 0)q.push(j);
		}
	}
}

int main() {
	cin >> n >> m;
	memset(dis,0x80,sizeof dis);	//初始化数组为负无穷(0x80)

	for (int i = 0; i < m; i++) {
		int a; Edge ee;
		cin >> a >> ee.to >> ee.w;
		e[a].push_back(ee);

		indegree[ee.to]++;	//记录入度出度
		outdegree[a]++;
	}

	queue<int>q1;

	//以下为排除除了1以外的入度为零的点
	//注意:必须要找到除1以外的入度为零的点而且把他们扩展出来的点也减少入度,不然会导致有些点在拓扑排序的时候没办法变为入度为零
	for (int i = 2; i <= n; i++) {
		if (indegree[i] == 0)q1.push(i);
	}
	while (q1.size()) {
		int t = q1.front();
		q1.pop();

		for (int i = 0; i < e[t].size(); i++) {
			int j = e[t][i].to;
			indegree[j]--;
			if (indegree[j] == 0)q1.push(j);
		}
	}

	topsort();	//拓扑排序

	if (dis[n] < -1e5)printf("-1");	//这里写-1e5是因为题目中要求的最小的边权为-1e5
	else printf("%d", dis[n]);

	return 0;
}
  • 9
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值