zoukankan      html  css  js  c++  java
  • Lunch Time

    hdu4807:http://acm.hdu.edu.cn/showproblem.php?pid=4807

    题意:给你n个点(0--n-1),点之间是有向边,0号点有k个人,现在0号点的k个人要到n-1号点。每条边有一个容量,就是单位时间内最多允许c个人通过,通过一条边需要一个单位时间,现在问你最后一个到达n-1号点最短的时间是多少。

    题解:这题用到网络流。怎么用呢?首先建图,边的容量自然是原来的容量,费用肯定为1.接着,我们可以想。先选一条路径的话,我们肯定选择一条费用最少的,这里就是距离最短的一条,因为,如果选择其他的一条,当那一条到达时,这一条已经到达,不会影响最后一个人的到达,只会贡献人数的减少。所以肯定选择费用最少的一条。然后,直选一条行不?答案是不一定的,因为这一条仅仅是费用最少,但是容量不一定是最大的。当第一个人到达的时候,接下来每一秒都会有y个人到达,y是流量。所以我们遍历下一条,选择下一条的时候,前一条肯要选,因为前一条不会影响这一条,并且会减少人数,提高运输量。所以,我们要遍历所有可行路径。s[i]表示到第i条时候第一次可以运输的总人数,则可以退出递推关系系s[i]=y[i] +(d[i]-d[i-1])*sum[i-1],y[i]表示第i条可行路径的流量,sum[i-1]表示前i-1条额流量和,d[i]表示第i条路径的费用。然后求费用流的时候更新就可以了。但是还要注意k==0时候要特判。

      1 #include<iostream>
      2 #include<cstdio>
      3 #include<cstring>
      4 #include<algorithm>
      5 #include<queue>
      6 #include<cmath>
      7 #define inf 100000000
      8 using namespace std;
      9 const int E=1000000;
     10 const int N=10002;
     11 struct Node{
     12     int v, cap, cost, next;    //  re记录逆边的下标。
     13 }edge[E];
     14 int n, m;
     15 long long ans,people;
     16 int k, head[N];
     17 int que[N], pre[N], dis[N];
     18 bool vis[N];
     19 void init(){//初始化
     20    k=ans=0;
     21    memset(head,-1,sizeof(head));
     22 }
     23 void addEdge(int u, int v, int ca, int co){
     24     edge[k].v = v;
     25     edge[k].cap = ca;
     26     edge[k].cost = co;
     27     edge[k].next = head[u];
     28     head[u] = k ++;
     29     edge[k].v = u;
     30     edge[k].cap = 0;
     31     edge[k].cost = -co;
     32     edge[k].next = head[v];
     33     head[v] = k ++;
     34 }
     35 bool spfa(){                  //  源点为0,汇点为n。
     36     int i;
     37     for(i = 1; i <= n; i ++){
     38         dis[i] = inf;
     39         vis[i] = false;
     40     }
     41     queue<int>Q;
     42     Q.push(1);
     43     dis[1]=0;
     44     vis[1] = true;
     45     while(!Q.empty()){       //  这里最好用队列,有广搜的意思,堆栈像深搜。
     46         int u = Q.front();
     47         Q.pop();
     48         for(i = head[u]; i != -1; i = edge[i].next){
     49             int v = edge[i].v;
     50             if(edge[i].cap && dis[v] > dis[u] + edge[i].cost){
     51                 dis[v] = dis[u] + edge[i].cost;
     52                 pre[v] = i;
     53                 if(!vis[v]){
     54                     vis[v] = true;
     55                    Q.push(v);
     56                 }
     57             }
     58         }
     59         vis[u] = false;
     60     }
     61     if(dis[n] == inf) return false;
     62     return true;
     63 }
     64 int  end(){
     65     int u, p, sum = inf;
     66     for(u = n; u != 1; u = edge[p^1].v){//0是超级源点
     67         p = pre[u];
     68         sum = min(sum, edge[p].cap);
     69     }
     70     for(u = n; u != 1; u = edge[p^1].v){
     71         p = pre[u];
     72         edge[p].cap -= sum;
     73         edge[p^1].cap += sum;
     74     }
     75     return sum;
     76 }
     77 int main(){
     78     int t1,t2,t3;
     79    while(~scanf("%d%d%d",&n,&m,&people)&&n>0){
     80         init();//初始化
     81        for(int i=1;i<=m;i++){
     82         scanf("%d%d%d",&t1,&t2,&t3);
     83           t1++;t2++;
     84           addEdge(t1,t2,t3,1);//无向图要建边两次
     85        }
     86        long long s=people,now=0,sum=0;
     87        ans=inf;
     88        while(spfa()){
     89           int y=end();
     90            s-=((long long)dis[n]-now)*sum+y;
     91            if(s<0)s=0;
     92            sum+=y;now=dis[n];
     93            long long temp=(long long)now+(long long)ceil(s*1.0/sum);
     94            if(temp<ans)ans=temp;
     95        }
     96        if(people==0)ans=0;
     97        if(ans==inf)printf("No solution
    ");
     98        else
     99        printf("%I64d
    ",ans);
    100    }
    101 }
    View Code
  • 相关阅读:
    package.json 字段全解析 share
    Git 使用方法
    浅谈移动前端的最佳实践(转)
    node启动服务后,窗口不能关闭。pm2了解一下
    WebSocket--客户端
    布局:多列等高布局方法
    uni-app调用支付宝、微信支付
    scss 入门基础
    mac生成iOS证书(配图)
    通过扫码打开IOS的App Store下载APP(Android版暂时没找到解决方法)
  • 原文地址:https://www.cnblogs.com/chujian123/p/3921563.html
Copyright © 2011-2022 走看看