-->

SCU 1095运送物资(最短路)

SCU 1095运送物资(最短路)

     X国发生了内战。起义军得到了广大人民的支持。在一次战役中,反动军队结集了大量兵力,围攻起义军的主堡W城。为支援前线,后方各个供给基地城市纷纷准备将物资运往W城。各基地及W城之间有的有公路相连。这就是说,有的基地不能将物资一次运到W城,必须通过中途的转运。

根据每条公路的长短和运送物资的多少,运送中将会有不同程度的损耗。现假设每条公路都有一个损耗系数,表示经过这条公路的物资总量与消耗量的比值。

另外,为保证物资安全到达,每个基地都会等所有要通过该基地转运的物资到齐后,连同本基地的物资一起,运到下一站。也就是说,从任何一个基地出发,都只能将物资运往另一基地,但允许多个基地的物资运往同一基地。

请编程预定出每个基地的运输路线,使到达W城的总物资最大。

输入:{input.txt}

第一行给出两个整数n(2<=n<=100)与m。其中n表示有n-1个基地(编号为1到n)与一个W城(编号为n);m表示有m条公路。

第二行给出了n-1个正整数(正整数<=50000),表示编号为1到n-1的基地要运送的物资数量。

接下来m行描述了m条公路的情况,每一行有3个数,如 1 2 0.1

表示1号城市与2号城市之间有一条公路连接,其损耗系数为0.1。

注意:数据给出的公路网,保证每个基地都能将物资运到W城。

输出:{ouput.txt}

共n行。

第一行是运到W城的最大物资数(结果保留两位小数)。

接下来n-1行分别有一个数,代表每个基地将物资运往的下一个基地或W城的编号。

样例输入

5 6

10 10 10 10

1 3 0

1 4 0

2 3 0

2 4 0

3 5 0

4 5 0

样例输出

40.00

3

3

5

5

解题报告

裸的最短路,在将边的权值表示为(1-损耗)的倒数。因为精度问题,倒数要开long double,并除以100,在计算路径时不是加而是乘。

 

#include<bits/stdc++.h>
#include<queue>
#include <algorithm>
#include<cstdio>
#include<cstring>
#include<cmath>
#define Pair pair<long double,int>
#define MAXN 1000+10
#define MAXM 600000+1
using namespace std;
int n,m,num,head[MAXN],s,t,pre[MAXN],v[MAXM];
double good[MAXN],ans;
long double dis[MAXN];
struct Edge{
    int next,to,exi,from;
    double dis;
}edge[MAXM];
void add(int from,int to,double dis)
{
    edge[++num].next=head[from];
    edge[num].to=to;
    edge[num].dis=dis;
    edge[num].from=from;
    head[from]=num;
}
void dij()
{
    memset(dis,0,sizeof(dis));
    memset(pre,0,sizeof(pre));
    memset(v,0,sizeof(v));
    priority_queue<Pair,vector<Pair>,greater<Pair> > h;
    for(int i=1;i<=n;i++) dis[i]=LONG_LONG_MAX;
    dis[s]=1*0.001;
    h.push(Pair(dis[s],s));
    while(h.size()>0)
    {
        int k=h.top().second;h.pop();
        if(v[k]) continue;
        v[k]=1;
        for(int i=head[k];i;i=edge[i].next)
        {    
            if(((dis[k]*edge[i].dis))<(dis[edge[i].to]))
            {
                dis[edge[i].to]=dis[k]*edge[i].dis;
                pre[edge[i].to]=edge[i].from;
                h.push(Pair(dis[edge[i].to],edge[i].to));
            }
        }
        
    }
}
int main()
{

    scanf("%d%d",&n,&m);
    for(int i=1;i<=n-1;i++)
        scanf("%lf",&good[i]);
    for(int i=1;i<=m;i++)
    {
        int x,y;double z;
        scanf("%d%d%lf",&x,&y,&z);
        add(x,y,1/(1-z));
        add(y,x,1/(1-z));
    }
    s=n;
    dij();
    for(int i=1;i<=n-1;i++)
         ans+=good[i]*(0.001/dis[i]);
    printf("%.2lf\n",ans);
    for(int i=1;i<=n-1;i++)
    {
        printf("%d\n",pre[i]);
    }
    
    return 0;
}

 

posted @ 2017-01-24 10:06  yangyaojia  阅读(258)  评论(0编辑  收藏  举报