PAT L3-011. 直捣黄龙

最短路。

先求出一个只包含最短路的$DAG$,然后在$DAG$上$dp$一下就可以了。英文变数字还有$map$转一下。

#include<map>
#include<set>
#include<ctime>
#include<cmath>
#include<queue>
#include<string>
#include<stack>
#include<vector>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<functional>
using namespace std;

int n,m,s,t,num[250];
char tmp[100],A[100],B[100];

map<string,int>xx;
map<int,string>yy;

vector<int>G[250],nG[250];
int Time[250][250];

int disS[250],disT[250],f[250];

int U[250*250],V[250*250],In[250];

int ss[250],pre[250],sl[250],dp[250];

void spfaS()
{
    memset(f,0,sizeof f);
    for(int i=1;i<=n;i++) disS[i]=0x7FFFFFFF;
    queue<int>Q; Q.push(s); f[s]=1; disS[s]=0;

    while(!Q.empty())
    {
        int h = Q.front(); Q.pop(); f[h]=0;
        for(int i=0;i<G[h].size();i++)
        {
            int to = G[h][i];
            if(disS[h]+Time[h][to]<disS[to])
            {
                disS[to] = disS[h]+Time[h][to];
                if(f[to]==0)
                {
                    f[to]=1;
                    Q.push(to);
                }
            }
        }
    }
}

void spfaT()
{
    memset(f,0,sizeof f);
    for(int i=1;i<=n;i++) disT[i]=0x7FFFFFFF;
    queue<int>Q; Q.push(t); f[t]=1; disT[t]=0;


    while(!Q.empty())
    {
        int h = Q.front(); Q.pop(); f[h]=0;
        for(int i=0;i<G[h].size();i++)
        {
            int to = G[h][i];
            if(disT[h]+Time[h][to]<disT[to])
            {
                disT[to] = disT[h]+Time[h][to];
                if(f[to]==0)
                {
                    f[to]=1;
                    Q.push(to);
                }
            }
        }
    }
}

void W()
{
    queue<int>Q; Q.push(s); dp[s]=1;

    while(!Q.empty())
    {
        int h = Q.front(); Q.pop();
        for(int i=0;i<nG[h].size();i++)
        {
            int to = nG[h][i];
            dp[to]+=dp[h];
            if(sl[h]+1>sl[to])
            {
                sl[to] = sl[h]+1;
                ss[to] = ss[h]+num[to];
                pre[to]=h;
            }

            else if(sl[h]+1==sl[to])
            {
                if(ss[h]+num[to]>ss[to])
                {
                    sl[to] = sl[h]+1;
                    ss[to] = ss[h]+num[to];
                    pre[to]=h;
                }
            }
            In[to]--;
            if(In[to]==0) Q.push(to);
        }
    }
}

int main()
{
    scanf("%d%d",&n,&m); scanf("%s%s",A,B);

    xx[A]=1; yy[1]=A;
    for(int i=2;i<=n;i++)
    {
        scanf("%s",tmp); xx[tmp]=i; yy[i]=tmp;
        scanf("%d",&num[i]);
    }
    s=1; t=xx[B];

    for(int i=1;i<=m;i++)
    {
        int P;
        scanf("%s%s",A,B); scanf("%d",&P);
        int u = xx[A], v = xx[B];

        U[i] = u; V[i]=v;
        Time[u][v]=Time[v][u]=P;
        G[u].push_back(v);
        G[v].push_back(u);
    }

    spfaS(); spfaT();

    for(int i=1;i<=m;i++)
    {
        if(disS[U[i]]+disT[V[i]]+Time[U[i]][V[i]]==disS[t])
        {
            nG[U[i]].push_back(V[i]);
            In[V[i]]++;
        }
        if(disS[V[i]]+disT[U[i]]+Time[U[i]][V[i]]==disS[t])
        {
            nG[V[i]].push_back(U[i]);
            In[U[i]]++;
        }
    }

    W();

    vector<int>Ans;
    stack<int>St;

    int now=t;
    while(1)
    {
        if(now==0) break;
        St.push(now);
        now=pre[now];
    }

    while(!St.empty())
    {
        Ans.push_back(St.top());
        St.pop();
    }

    for(int i=0;i<Ans.size();i++)
    {
        cout<<yy[Ans[i]];
        if(i<Ans.size()-1) printf("->");
        else printf("\n");
    }

    printf("%d %d %d\n",dp[t],disS[t],ss[t]);

    return 0;
}

 

posted @ 2017-03-22 16:05  Fighting_Heart  阅读(287)  评论(0编辑  收藏  举报