题目没给出数据范围...因为其他最短路方法掌握的不熟练就果断使用了floyd...
比较麻烦的是要输出路径,如果最短路不止一个输出字典树最小的...字典树不是问题...因为感觉floyd从小到大遍历字典树顺序不是问题,但是记录路径方法不是太对,一直WA..最后的程序还是加上了对字典树的判断.AC
path[i][j]记录的是从i点到j点之间最短路的经过的第一个节点。
#include<stdio.h>
#include<string.h>
int g[101][101];
int dis[101][101];
int path[101][101],b[101];
#define inf 9999999
void floyd(int n)
{
int i,j,k,sum;
for(i=1;i<=n;i++)
{
for(j=1;j<=n;j++)
{
dis[i][j]=g[i][j];
path[i][j]=j;
}
}
for(k=1;k<=n;k++)
{
for(i=1;i<=n;i++)
for(j=1;j<=n;j++)
{
sum=dis[i][k]+dis[k][j]+b[k];
if(dis[i][j]>sum)
{
dis[i][j]=sum;
path[i][j]=path[i][k];
}
else if(path[i][j]>path[i][k]&&dis[i][j]==sum)
path[i][j]=path[i][k];
}
}
}
int main()
{
int n,i,j;
while(scanf("%d",&n),n)
{
for(i=1;i<=n;i++)
{
for(j=1;j<=n;j++)
{
scanf("%d",&g[i][j]);
if(g[i][j]==-1)
g[i][j]=inf;
}
}
for(i=1;i<=n;i++)
scanf("%d",&b[i]);
floyd(n);
int s,e;
while(scanf("%d %d",&s,&e),s!=-1||e!=-1)
{
int x=s;
printf("From %d to %d :\nPath: %d",s,e,s);
while(x!=e)
{
printf("-->%d",path[x][e]);
x=path[x][e];
}
printf("\nTotal cost : %d\n",dis[s][e]);
printf("\n");
}
}
return 0;
}