你来到一个迷宫前。该迷宫由若干个房间组成,每个房间都有一个得分,第一次进入这个房间,你就可以得到这个分数。还有若干双向道路连结这些房间,你沿着这些道路从一个房间走到另外一个房间需要一些时间。游戏规定了你的起点和终点房间,你首要目标是从起点尽快到达终点,在满足首要目标的前提下,使得你的得分总和尽可能大。现在问题来了,给定房间、道路、分数、起点和终点等全部信息,你能计算在尽快离开迷宫的前提下,你的最大得分是多少么?
收起
输入
第一行4个整数n (<=500), m, start, end。n表示房间的个数,房间编号从0到(n - 1),m表示道路数,任意两个房间之间最多只有一条道路,start和end表示起点和终点房间的编号。 第二行包含n个空格分隔的正整数(不超过600),表示进入每个房间你的得分。 再接下来m行,每行3个空格分隔的整数x, y, z (0<z<=200)表示道路,表示从房间x到房间y(双向)的道路,注意,最多只有一条道路连结两个房间, 你需要的时间为z。 输入保证从start到end至少有一条路径。
输出
一行,两个空格分隔的整数,第一个表示你最少需要的时间,第二个表示你在最少时间前提下可以获得的最大得分。
输入样例
3 2 0 2 1 2 3 0 1 10 1 2 11
输出样例
21 6
#include<stdio.h>
#include<string.h>
#include<algorithm>
using namespace std;
#define inf 0x3f3f3f3f
int val[520];
int mp[510][510];
int dis[510];//时间
bool vis[510];
int dd[510];//结点到开始符号的最大值
void djk(int n,int st,int ed)
{
dd[st]=val[st];//
memset(vis,false,sizeof(vis));
// memset(dd,0,sizeof(dd));
for(int i=0;i<n;i++)
{
dis[i]=mp[st][i];
if(mp[st][i]!=inf) //这里注意
dd[i]=dd[st]+val[i];
}
vis[st]=1;
dis[st]=0;
for(int k=1;k<n;k++)
{
int mn=inf,id=-1;
for(int i=0;i<n;i++)
{
if(!vis[i]&&(mn>dis[i]))
{
mn=dis[i];
id=i;
}
}
vis[id]=1;
for(int i=0;i<n;i++)
{
if(!vis[i]&&mp[id][i]!=inf)
{
if(dis[i]>dis[id]+mp[id][i])
dis[i]=dis[id]+mp[id][i],dd[i]=dd[id]+val[i];//同时更新
}
}
for(int i=0;i<n;i++)
{
if(dis[i]==dis[id]+mp[id][i])
dd[i]=max(dd[i],dd[id]+val[i]);//寻找最大
}
}
printf("%d %d\n",dis[ed],dd[ed]);
}
int main()
{
int st,ed,n,m;
memset(mp,inf,sizeof(mp));
scanf("%d%d%d%d",&n,&m,&st,&ed);
for(int i=0;i<n;i++)
{
scanf("%d",&val[i]);
}
for(int i=0;i<m;i++)
{
int x,y,z;
scanf("%d%d%d",&x,&y,&z);
if(mp[x][y]>z)
mp[x][y]=z,mp[y][x]=z;
}
djk(n,st,ed);
return 0;
}