zoukankan      html  css  js  c++  java
  • P2176 [USACO14FEB]路障Roadblock

    题目传送门

    十分值得一做的最短路,题目意思十分明确,一条边权值加倍后最多比加倍前的最短路花费多多少。首先看到m<=5000,第一念头就是跑m遍最短路,但是会严重超时,实际上是由于有些边的权值改变,对最短路没有造成任何影响,才导致了我们程序的严重超时,所以我们采取第一次跑最短路记路径的方法。开三个辅助数组path[]、pre[]和edge[],path[]就是用来记最短路经过了哪些编号的边(邻接表存边),pre[i]表示走i号节点之前走的边,edge[i]表示到i号节点之前所经过的点,然后while循环逆推路径并记录,最后依次枚举最短路经过的边*2即可

    算法步骤:

    1)读入,邻接表存边,注意这里一定要用邻接表,不然后边不好记路径

    2)先跑一遍SPFA,计算pre和edge数组,并求出“裸”的最短路

    3)用while循环推导path数组

    4)依次将path数组对应的每条边*2,并求出差值,打擂台取最大差值(计算完之后不要忘记除回来)

    5)输出即可

    参考程序如下:

     1 #include<iostream>
     2 #include<cstring>
     3 #include<queue>
     4 using namespace std;
     5 int ans,maxx,n,m,x,y,z,v[100001],w[100001],head[100001],nxt[100001],cnt,path[100001],dist[100001],pre[100001],now,num,edge[100001];
     6 bool vis[100001];
     7 void add(int a,int b,int c)
     8 {
     9     v[++cnt]=b;
    10     w[cnt]=c;
    11     nxt[cnt]=head[a];
    12     head[a]=cnt;
    13 }
    14 void spfa(int s)
    15 {
    16     memset(dist,20,sizeof(dist));
    17     queue<int>q;
    18     q.push(s);
    19     vis[s]=1;
    20     dist[s]=0;
    21     while(!q.empty())
    22     {
    23         int t=q.front();
    24         q.pop();
    25         vis[t]=0;
    26         for(int i=head[t];i;i=nxt[i])
    27         {
    28             int y=v[i];
    29             if(dist[y]>dist[t]+w[i])
    30             {
    31                 dist[y]=dist[t]+w[i];
    32                 pre[y]=i;edge[y]=t;
    33                 if(!vis[y])
    34                 {
    35                     q.push(y);
    36                     vis[y]=1;
    37                 }
    38             }
    39         }
    40     }
    41 }
    42 int main()
    43 {
    44     cin>>n>>m;
    45     for(int i=1;i<=m;i++)
    46     {
    47         cin>>x>>y>>z;
    48         add(x,y,z);
    49         add(y,x,z);
    50     }
    51     spfa(1);
    52     ans=dist[n];
    53     now=n;
    54     while(now!=1)//记录路径,本程序中最难理解的地方,请务必好好理解!!! 
    55     {
    56         path[++num]=pre[now];
    57         now=edge[now];
    58     }
    59     for(int i=1;i<=num;i++)
    60     {
    61         w[path[i]]*=2;
    62         spfa(1);
    63         maxx=max(maxx,dist[n]);
    64         w[path[i]]/=2;
    65     }
    66     cout<<maxx-ans<<endl;
    67     return 0;
    68 }
    View Code

      

  • 相关阅读:
    HNOI2003 消防局的设立
    APIO2007 风铃
    SDOI2006 保安站岗
    消息传递
    [POI2008]STA-Station
    JLOI2015 城池攻占
    BOI2004 sequence
    Monkey King
    APIO2012 Dispatching
    HTML meta 文本 格式排版 链接图表 列表 表单 frame后台布局实例
  • 原文地址:https://www.cnblogs.com/szmssf/p/10989332.html
Copyright © 2011-2022 走看看