小W旅遊railway

0
1


對於一家鐵路公司,我們可以首先使用 Floyd 算法求出任
意兩點 x, y 間只經過屬於該家鐵路公司鐵路的最短路,那麼在新
圖中我們在 x, y 間加一條 x 到 y 最短路對應的花費爲邊權的邊。
接下來只要在新圖中使用 Floyd 算法求出任意兩點間的最
小花費就可以了。
跟昨天寫的<跑路>有點像
http://blog.csdn.net/kenxhe/article/details/53072146

總結一下, floyd的優勢在於可以求出任意兩個點之間的距離, 可利用這種特性來構建新圖, 解決問題.

代碼不是我的, 將就着看吧

#include<cstdio>
#include<cstring>
#include<algorithm>

using namespace std;
const int N=1010;
int p[N],q[N][N],r[N][N],f[N][N][21],ans,cost[20010][21];
int n,m,tot,x,y,z,k,kk,s,t,dis[N][N],c[N][N],fa[N];

int fin(int x)
{
    if (fa[x]!=x)
        fa[x]=fin(fa[x]);
    return fa[x];
}

int main()
{
    memset(f,0x16,sizeof(f));
    scanf("%d%d%d%d%d",&n,&m,&k,&s,&t);
    for (int i=1;i<=n;i++) fa[i]=i;
    for (int i=1;i<=m;i++)
    {
        scanf("%d%d%d%d",&x,&y,&z,&kk);
        f[x][y][kk]=z;
        c[x][y]=kk;
        f[y][x][kk]=z;
        c[y][x]=kk;
        int fx=fin(x),fy=fin(y);
        if (fx!=fy)
            fa[fx]=fy;
    }

    int fs=fin(s),ft=fin(t);
    if (fs!=ft)
    {
        printf("-1\n");
        return 0;
    }

    for (int i=1;i<=k;i++)
        scanf("%d",p+i);
    for (int i=1;i<=k;i++)
    {
        for (int j=1;j<p[i];j++)
            scanf("%d",&q[i][j]);
        for (int j=1;j<=p[i];j++)
            scanf("%d",&r[i][j]);
    }

    for (int i=1;i<=k;i++)
        for (int j=1;j<=20000;j++)
        {
            int ii,ans=0;
            for (ii=1;ii<p[i];ii++)
            {
                if (j<q[i][ii]) break;
                ans+=(q[i][ii]-q[i][ii-1])*r[i][ii];
            }
            if (j>q[i][ii-1])
                ans+=(j-q[i][ii-1])*r[i][ii];
            cost[j][i]=ans;
        }
    for (int cc=1;cc<=k;++cc)
    for (int kk=1;kk<=n;kk++)
        for (int i=1;i<=n;i++) if (i!=kk)    
            for (int j=1;j<=n;j++) if (i!=j)
                f[i][j][cc]=min(f[i][j][cc],f[i][kk][cc]+f[kk][j][cc]);
//              if (c[i][kk]==c[kk][j])
//                  f[i][j][c[kk][j]]=min(f[i][j][c[kk][j]],f[i][kk][c[i][kk]]+f[kk][j][c[kk][j]]);

    for (int i=1;i<=n;i++)
        for (int j=1;j<=n;j++) if (i!=j)
            for (int col=1;col<=k;col++)
            if (f[i][j][col]<20000000)
                f[i][j][col]=cost[f[i][j][col]][col];

    memset(dis,0x16,sizeof(dis));
    for (int i=1;i<=n;i++)
        for (int j=1;j<=n;j++) if (i!=j)
            for (int col=1;col<=k;col++) 
                dis[i][j]=min(dis[i][j],f[i][j][col]);

    for (int kk=1;kk<=n;kk++) 
        for (int i=1;i<=n;i++) if (i!=kk)
            for (int j=1;j<=n;j++) if (i!=j)
                dis[i][j]=min(dis[i][j],dis[i][kk]+dis[kk][j]);
    printf("%d\n",dis[s][t]);
}
發佈了57 篇原創文章 · 獲贊 0 · 訪問量 2萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章