HDU 4725 The Shortest Path in Nya Graph-【SPFA最短路】

题目:http://acm.hdu.edu.cn/showproblem.php?pid=4725


题意:有N个点和N层..一层有X个点(0<=X<=N).两邻两层间有一条路花费C。还有M条小路在两个点之间。问从第一个点走到第N个点最短路是多少...


题解:依然是在点之间SPFA。不过在跑的时候不仅要跑与当前点相连接的点。还有把当前点所在层的相邻层的点判断是否加入队列...


CODE:

#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
#include<vector>
#include<queue>

#define mkp make_pair
#define fst first
#define scd second

using namespace std;
int dis[100011];
int vis[100011];
int lay[100011];
vector<int>vec[100011];
struct Edge_t{
    int to,next,cap;
}edge[500000];
int head[100011],et;

inline void adde(int u,int v,int w){
    edge[et].to=v;
    edge[et].cap=w;
    edge[et].next=head[u];
    head[u]=et++;
}

inline void spfa(int n,int C){
    queue<int>q;
    q.push(1);
    memset(vis,0,sizeof vis);
    memset(dis,-1,sizeof dis);
    dis[1]=0;
    int e,u,v,size,i,k;
    while(!q.empty()){
        u=q.front();q.pop();
        vis[u]=0;
        for(e=head[u];~e;e=edge[e].next){
            v=edge[e].to;
            if(dis[v]<0 || dis[v]>dis[u]+edge[e].cap){
                dis[v]=dis[u]+edge[e].cap;
                if(!vis[v]){
                    vis[v]=1;
                    q.push(v);
                }
            }
        }
        if(lay[u]>1){
            size=vec[k=(lay[u]-1)].size();
            for(i=0;i<size;++i){
                v=vec[k][i];
                if(dis[v]<0 || dis[v]>dis[u]+C){
                    dis[v]=dis[u]+C;
                    if(!vis[v]){
                        vis[v]=1;
                        q.push(v);
                    }
                }
            }
        }
        if(lay[u]<n){
            size=vec[k=lay[u]+1].size();
            for(i=0;i<size;++i){
                v=vec[k][i];
                if(dis[v]<0 || dis[v]>dis[u]+C){
                    dis[v]=dis[u]+C;
                    if(!vis[v]){
                        vis[v]=1;
                        q.push(v);
                    }
                }
            }
            
        }
    }
}

int main(){
    int t,tt=0,C;
    int n,m,u,v,i,size,w;
    scanf("%d",&t);
    while(t--){
        scanf("%d%d%d",&n,&m,&C);
        memset(head,-1,sizeof head);et=0;
        memset(vis,0,sizeof vis);
        for(i=1;i<=n;++i){
            scanf("%d",&lay[i]);//第i个点所以层为lay[i]
            vec[i].clear();
        }
        while(m--){
            scanf("%d%d%d",&u,&v,&w);
            adde(u,v,w);adde(v,u,w);
            if(!vis[u]){//把点加入层..
                vec[lay[u]].push_back(u);
                vis[u]=1;
            }
            if(!vis[v]){
                vec[lay[v]].push_back(v);
                vis[v]=1;
            }
        }
        if(!vis[1])//始点和终于一定要在某一层内..
            vec[lay[1]].push_back(1);
        if(!vis[n])
            vec[lay[n]].push_back(n);
        for(i=1;i<=n;++i)//如果某一层没有点
			if(vec[lay[i]].empty())
                vec[lay[i]].push_back(i);
        spfa(n,C);
        printf("Case #%d: %d
",++tt,dis[n]);
    }
    return 0;
}


原文地址:https://www.cnblogs.com/suncoolcat/p/3320129.html