[BZOJ2763][JLOI2011]飞行路线

Time Limit: 10 Sec
Memory Limit: 128 MB

Description
Alice和Bob现在要乘飞机旅行,他们选择了一家相对便宜的航空公司。该航空公司一共在n个城市设有业务,设这些城市分别标记为 0 0 0 n − 1 n-1 n1,一共有 m m m种航线,每种航线连接两个城市,并且航线有一定的价格。Alice和Bob现在要从一个城市沿着航线到达另一个城市,途中可以进行转机。航空公司对他们这次旅行也推出优惠,他们可以免费在最多 k k k种航线上搭乘飞机。那么Alice和Bob这次出行最少花费多少?
Input
数据的第一行有三个整数, n , m , k n,m,k n,m,k,分别表示城市数,航线数和免费乘坐次数。
第二行有两个整数, s , t s,t s,t,分别表示他们出行的起点城市编号和终点城市编号。 ( 0 &lt; = s , t &lt; n ) (0&lt;=s,t&lt;n) (0<=s,t<n)
接下来有 m m m行,每行三个整数, a , b , c a,b,c a,b,c,表示存在一种航线,能从城市 a a a到达城市 b b b,或从城市 b b b到达城市 a a a,价格为 c c c。( 0 &lt; = a , b &lt; n , a 0&lt;=a,b&lt;n,a 0<=a,b<n,a b b b不相等, 0 &lt; = c &lt; = 1000 0&lt;=c&lt;=1000 0<=c<=1000)

Output

只有一行,包含一个整数,为最少花费。

Sample Input

5 6 1
0 4
0 1 5
1 2 5
2 3 5
3 4 5
2 3 3
0 2 100

Sample Output

8

HINT

对于30%的数据, 2 &lt; = n &lt; = 50 , 1 &lt; = m &lt; = 300 , k = 0 2&lt;=n&lt;=50,1&lt;=m&lt;=300,k=0 2<=n<=50,1<=m<=300,k=0
对于50%的数据, 2 &lt; = n &lt; = 600 , 1 &lt; = m &lt; = 6000 , 0 &lt; = k &lt; = 1 2&lt;=n&lt;=600,1&lt;=m&lt;=6000,0&lt;=k&lt;=1 2<=n<=600,1<=m<=6000,0<=k<=1
对于100%的数据, 2 &lt; = n &lt; = 10000 , 1 &lt; = m &lt; = 50000 , 0 &lt; = k &lt; = 10 2&lt;=n&lt;=10000,1&lt;=m&lt;=50000,0&lt;=k&lt;=10 2<=n<=10000,1<=m<=50000,0<=k<=10

题解:
裸的分层图最短路

#include<bits/stdc++.h>
#define LiangJiaJun main
#define INF 1999122700
#define ppa pair<int,pair<int,int> >
using namespace std;
int ne,h[10004];
struct edge{
    int to,nt,w;
}e[2000004];
int n,k,m;
int dis[14][10004];
bool vis[14][10004];
priority_queue<ppa,vector<ppa>,greater<ppa> >q;
void add(int u,int v,int w){
     e[++ne].to=v;e[ne].w=w;e[ne].nt=h[u];
     h[u]=ne;
}
int dijkstra(int S,int T){
    for(int i=0;i<=k;i++)
        for(int j=1;j<=n;j++)
            dis[i][j]=INF;
    for(int i=0;i<=k;i++){
        dis[i][S]=0;
        q.push(make_pair(0,make_pair(i,S)));
    }
    while(!q.empty()){
        int dg=q.top().second.first,x=q.top().second.second;
        q.pop();
        if(vis[dg][x])continue;
        vis[dg][x]=1;
        for(int i=h[x];i;i=e[i].nt){
            if( dis[dg][e[i].to]>dis[dg][x]+e[i].w){
                dis[dg][e[i].to]=dis[dg][x]+e[i].w;
                q.push(make_pair(dis[dg][e[i].to],make_pair(dg,e[i].to)));
            }
            if(dg>=k)continue;
            if( dis[dg+1][e[i].to]>dis[dg][x]){
                dis[dg+1][e[i].to]=dis[dg][x];
                q.push(make_pair(dis[dg+1][e[i].to],make_pair(dg+1,e[i].to)));
            }
        }
    }
    int ans=dis[0][T];
    for(int i=0;i<=k;i++)ans=min(ans,dis[i][T]);
    return dis[k][T];
}
int w33ha(){
    ne=0;
    int S,T;
    scanf("%d%d",&S,&T);
    ++S;++T;
    memset(vis,0,sizeof(vis));
    memset(h,0,sizeof(h));
    for(int i=1;i<=m;i++){
        int u,v,w;
        scanf("%d%d%d",&u,&v,&w);
        ++u;++v;
        add(u,v,w);
        add(v,u,w);
    }
    int res=dijkstra(S,T);
    printf("%d\n",res);
    return 0;
}
int LiangJiaJun(){
    while(scanf("%d%d%d",&n,&m,&k)!=EOF)w33ha();
    return 0;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值