题目描述
有一个邮递员要送东西,邮局在节点 11。他总共要送 n-1n−1 样东西,其目的地分别是节点 22 到节点 nn。由于这个城市的交通比较繁忙,因此所有的道路都是单行的,共有 mm 条道路。这个邮递员每次只能带一样东西,并且运送每件物品过后必须返回邮局。求送完这 n-1n−1 样东西并且最终回到邮局最少需要的时间。
输入格式
第一行包括两个整数,nn 和 mm,表示城市的节点数量和道路数量。
第二行到第 (m+1)(m+1) 行,每行三个整数,u,v,wu,v,w,表示从 uu 到 vv 有一条通过时间为 ww 的道路。
输出格式
输出仅一行,包含一个整数,为最少需要的时间。
输入输出样例
输入 #1复制
5 10 2 3 5 1 5 5 3 5 6 1 2 8 1 3 8 5 3 4 4 1 8 4 5 3 3 5 6 5 4 2输出 #1复制
83说明/提示
对于 30\%30% 的数据,1 \leq n \leq 2001≤n≤200。
对于 100\%100% 的数据,1 \leq n \leq 10^31≤n≤103,1 \leq m \leq 10^51≤m≤105,1\leq u,v \leq n1≤u,v≤n,1 \leq w \leq 10^41≤w≤104,输入保证任意两点都能互相到达。
//此题要得到每两个点之间的最短权值,所以要用floyd
//开o2才能过,否则会t(这是个什么原理,好神奇)
因为输入的数据中,两点之间的距离可能不止一个,为避免覆盖,一定要去重边,选择多次输入中最小的距离作为该两点之间的初始距离
因为邮递员每送完一封信还要回到邮局(初始点),又因为floyd算出了每两点之间的最短路径,所以最终计算结果时直接在for循环中ans+=v[1][i]+v[i][1];即可
ac代码(要开O2优化)
#include<bits/stdc++.h>
using namespace std;
#define inf 999999
int n,m,v[1010][1010];
int x,y,z;
int main()
{
memset(v,inf,sizeof(v));
scanf("%d%d",&n,&m);
for(int i=1; i<=m; ++i)
{
scanf("%d%d%d",&x,&y,&z);
v[x][y]=min(z,v[x][y]);//去重边
}
//flody
for(int k=1; k<=n; ++k)
for(int i=1; i<=n; ++i)
for(int j=1; j<=n; ++j)
v[i][j]=min(v[i][k]+v[k][j],v[i][j]);
int ans=0;
for(int i=2; i<=n; ++i)
ans+=v[1][i]+v[i][1];//每次去的时间+回来的时间
cout<<ans;
return 0;
}