zoukankan      html  css  js  c++  java
  • UVA 11374 Airport Express(最短路)

    最短路。

    把题目抽象一下:已知一张图,边上的权值表示长度。现在又有一些边,只能从其中选一条加入原图,使起点->终点的距离最小。

    当加上一条边a->b,如果这条边更新了最短路,那么起点st->终点ed的最小距离=st->a  +  a->b  +b->ed 三个值的最短距离之和。于是正反求两次单元最短路。再将k条边遍历一遍就好了。

    最近在改代码风格,写起来很别扭。。uva又挂了,最让我不理解的是http://www.cnblogs.com/arbitrary/archive/2013/02/06/2908099.html这个家伙的代码竟然能ac,问题是我稍微改一下就submission error,难道代码还能防盗= =

      1 #include<cstdio>
      2 #include<cstring>
      3 #include<vector>
      4 #include<queue>
      5 #include<algorithm>
      6 #define clr(a,x) memset(a,x,sizeof(a));
      7 #define ref(i,a,b) for(int i=a;i<=b;i++)
      8 using namespace std;
      9 
     10 const int MAXN=1111;
     11 const int INF =55555;
     12 
     13 struct Edge{
     14     int u,v,c;
     15 };
     16 
     17 struct Node{
     18     int d,u;
     19     bool operator <(const Node x)const {
     20         return d>x.d;
     21     }
     22 };
     23 
     24 struct Dij{
     25     int n,m;
     26     vector<Edge>edge;
     27     vector<int>G[MAXN];
     28     bool vis[MAXN];
     29     int d[MAXN];
     30     int p[MAXN];
     31 
     32     void init(int n)
     33     {
     34         this->n=n;
     35         ref(i,0,n-1)G[i].clear();
     36         edge.clear();
     37     }
     38 
     39     void add(int u,int v,int c)
     40     {
     41         edge.push_back((Edge){u,v,c});
     42         m=edge.size();
     43         G[u].push_back(m-1);
     44     }
     45 
     46     void dij(int st)
     47     {
     48         priority_queue<Node>q;
     49         ref(i,0,n-1)d[i]=(i==st)?0:INF;
     50         clr(vis,0);
     51         q.push((Node){0,st});
     52         while(!q.empty())
     53         {
     54             Node x=q.top();
     55             q.pop();
     56             int u=x.u;
     57             if(vis[u])
     58                 continue;
     59             vis[u]=true;
     60             ref(i,0,G[u].size()-1){
     61                 Edge e=edge[G[u][i]];
     62                 if(d[e.v]>d[u]+e.c){
     63                     d[e.v]=d[u]+e.c;
     64                     p[e.v]=G[u][i];
     65                     q.push((Node){d[e.v],e.v});
     66                 }
     67             }
     68         }
     69     }
     70 };
     71 
     72 Dij solver;
     73 int d1[MAXN],d2[MAXN];
     74 int p[MAXN];
     75 
     76 int main()
     77 {
     78     int cnt=0;
     79     int n,m,k,st,ed;
     80     while(~scanf("%d%d%d",&n,&st,&ed))
     81     {
     82         if(cnt++)
     83             puts("");
     84 
     85         st--;ed--;
     86         solver.init(n);
     87 
     88         int u,v,c;
     89         scanf("%d",&m);
     90         ref(i,0,m-1){
     91             scanf("%d%d%d",&u,&v,&c);
     92             u--;v--;
     93             solver.add(u,v,c);
     94             solver.add(v,u,c);
     95         }
     96 
     97         solver.dij(st);
     98         ref(i,0,n-1)
     99             d1[i]=solver.d[i];
    100 
    101         solver.dij(ed);
    102         ref(i,0,n-1){
    103             d2[i]=solver.d[i];
    104             p[i]=solver.p[i];
    105         }
    106         p[ed]=-1;
    107 
    108         scanf("%d",&k);
    109         int s=d1[ed],a,b;
    110         a=b=-1;
    111         ref(i,0,k-1){
    112             scanf("%d%d%d",&u,&v,&c);
    113             u--;v--;
    114             if(s>d1[u]+c+d2[v]){
    115                 s=d1[u]+c+d2[v];
    116                 a=u;b=v;
    117             }
    118             if(s>d1[v]+c+d2[u]){
    119                 s=d1[v]+c+d2[u];
    120                 a=v;b=u;
    121             }
    122         }
    123         
    124         int flog=0;
    125         if(a==-1){       //不乘商务车
    126             int x=st;
    127 
    128             while(x!=ed)
    129             {
    130                 if(!flog){printf("%d",x+1);flog=1;}
    131                 else printf(" %d",x+1);
    132                 x=solver.edge[p[x]].u;       //p[]中保存的是 u->v 这条边的序号
    133             }
    134             printf(" %d",ed+1);
    135             printf("
    Ticket Not Used
    %d
    ",s);
    136         }else{        //乘坐商务车
    137             int x=st;
    138             while(x!=a)
    139             {
    140                 if(!flog){printf("%d",x+1);flog=1;}
    141                 else printf(" %d",x+1);
    142                 x=solver.edge[p[x]].u;
    143             }
    144             if(x!=st)printf(" %d",x+1);
    145             else printf("%d",x+1);
    146             x=b;
    147             while(x!=ed)
    148             {
    149                 printf(" %d",x+1);
    150                 x=solver.edge[p[x]].u;
    151             }
    152             printf(" %d
    ",ed+1);
    153             printf("%d
    %d
    ",a+1,s);
    154         }
    155     }
    156     return 0;
    157 }
    158 
    159 /*
    160 4 1 4
    161 4
    162 1 2 2
    163 1 3 3
    164 2 4 4
    165 3 4 5
    166 1
    167 1 4 7
    168 
    169 2 1 2
    170 1
    171 1 2 2
    172 1
    173 1 2 1
    174 */
    View Code
  • 相关阅读:
    【习题整理】分块+莫队(未完待续)
    【bzoj4198】【Noi2015】荷马史诗
    【bzoj2006】【NOI2015】超级钢琴
    【bzoj1029】【JSOI2007】建筑抢修
    【bzoj1483】【HNOI2009】梦幻布丁
    【bzoj4195】【NOI2015】程序自动分析
    Rearrangement inequality
    hdu1047
    hdu1046
    hdu1045
  • 原文地址:https://www.cnblogs.com/zstu-abc/p/3250705.html
Copyright © 2011-2022 走看看