#sPFAybttg030305. 1508:Easy SSSP

1508:Easy SSSP

1508:Easy SSSP

题目描述

输入数据给出一个有 NN 个节点,MM 条边的带权有向图。要求你写一个程序,判断这个有向图中是否存在负权回路。如果从一个点沿着某条路径出发,又回到了自己,而且所经过的边上的权和小于 0,就说这条路是一个负权回路。

如果存在负权回路,只输出一行 -1;如果不存在负权回路,再求出一个点 SS 到每个点的最短路的长度。约定:SSSS 的距离为 00,如果 SS 与这个点不连通,则输出 NoPath

输入格式

第一行三个正整数,分别为点数 NN,边数 MM,源点 SS

以下 MM 行,每行三个整数 a,b,ca, b, c,表示点 aa 到点 bb 之间连有一条有向边,权值为 cc

输出格式

如果存在负权环,只输出一行 -1,否则按以下格式输出:

NN 行,第 ii 行描述 SS 点到点 ii 的最短路:

  • 如果 SSii 不连通,输出 NoPath
  • 如果 i=Si=S,输出 00
  • 其他情况输出 SSii 的最短路的长度。

样例

样例输入 #1

6 8 1
1 3 4
1 2 6
3 4 -7
6 4 2
2 4 5
3 6 3
4 5 1
3 5 4

样例输出 #1

0
6
4
-3
-2
7

样例解释 #1

  • 图中没有负权环。
  • 从源点 S=1S=1 出发:
    • 1100
    • 22121 \to 2,长度 66
    • 33131 \to 3,长度 44
    • 441341 \to 3 \to 4,长度 4+(7)=34 + (-7) = -3
    • 5513451 \to 3 \to 4 \to 5,长度 4+(7)+1=24 + (-7) + 1 = -2
    • 661361 \to 3 \to 6,长度 4+3=74 + 3 = 7

数据范围

对于全部数据:

  • 2N10002 \le N \le 1000
  • 1M1051 \le M \le 10^5
  • 1a,b,SN1 \le a, b, S \le N
  • c106|c| \le 10^6

时空限制

  • 时间限制:1000 ms
  • 内存限制:65536 KB

注意:本题需要先判断整个图中是否存在负权环(可以用 Bellman-Ford 或 SPFA 判断),如果没有负权环,再计算从源点 SS 到所有点的最短路径(注意处理不连通的情况)。由于 N1000N \le 1000M105M \le 10^5,需要选择高效的算法。