题目大意:
解题思路:
首先我们先明确为什么会出现无穷多条路径
- 这个答案显而易见,就是有0环
那么怎样的0环会使得答案出现无穷多条路径呢
- 对于 k = 1 k=1 k=1的情况,作如下图:
- 因为最短路为3, k = 1 k=1 k=1,所以可行的路径大小在 [ 3 , 4 ] [3,4] [3,4],那么处于下方的0环就不在我们考虑的范围内,而上方的0环则会使我们的路径有无限多条
- 所以当且仅当0边处在可走区间的范围内时,这些0边构成的0环才会对结果产生影响,同时我们把这些0边又叫做可行0边
那么如何把得到所有的可行边,再把这些可行边重新建图(注意是可行边,不单单指可行0边)
- 此处借由大佬博客的一句话:
判断是否可行的方法为如果某条边连接的两个端点分别到起点和终点的距离加上此条边的长度小于等于起点到终点的最短路+k,说明通过该条路径存在答案,并判断其是否在以1为源的最短路上,将其加入到新图(即最短路边集) - 这样子既能保证所有可行的0边加入到新图(因为所有边权非负,所以可行的0边连接的两端恰好满足以上两个条件),又能保证在该图上跑dp时不会使最短路有增量(此处后面会有解释)
- 上图就会变成:
- 接下来只要利用拓扑排序判环即可(想想为什么如果有环一定是0环,再看看上面两个判断条件)
接下来就是dp在分层图上求解答案的过程
- 设 d p [ i ] [ j ] dp[i][j] dp[i][j]代表 1 1 1到 i i i结点增量为 j j j的所有可行边
- 设想有 k + 1 k+1 k+1层图,那么每一层得到的解即代表不同增量下的解,最后算总和即可
- 而如何在某一层跑dp时不会影响到其他层呢(疯狂暗示,看看上面),其实只要在一开始建的最短路边集上跑就行啦
- 那么层与层之间是否毫无关联呢,显然不是呀!!!
如果毫无关联的话,那不就相当于k始终为0,k显然不会始终为0 - 那么每一层之间的关系,就由那些没有加入最短路边集的边来联系,因为跑它们,会有增量,所以就能用来联系每一层
然后,每条边的增量又要重新计算(可看下面代码) - 最后累加dp[n][0~k]即可
AC代码
#include <bits/stdc++.h>
#define inf 1000000000
using