zoukankan      html  css  js  c++  java
  • 测试3T3-差分约束路径

    问题 C: 种树

    时间限制: 2 Sec  内存限制: 128 MB
    提交: 97  解决: 36
    [提交][状态][讨论版]

    题目描述

    为了绿化乡村,H村积极响应号召,开始种树了。

    H村里有n幢房屋,这些屋子的排列顺序很有特点,在一条直线上。于是方便起见,我们给它们标上1~n。树就种在房子前面的空地上。

    同时,村民们向村长提出了m个意见,每个意见都是按如下格式:希望第li个房子到第ri个房子的房前至少有ci棵树。

    因为每个房屋前的空地面积有限,所以每个房屋前最多只能种ki棵树。

    村长希望在满足村民全部要求的同时,种最少的树以节约资金。请你帮助村长。

     

     

     

     

    输入

    输入文件名为tree.in

    输入第1行,包含两个整数nm

    第2行,有n个整数ki

    第2~m+1行,每行三个整数lirici

     

     

     

     

    输出

    输出文件名为tree.out

    输出1个整数表示在满足村民全部要求的情况下最少要种的树。村民提的要求是可以全部满足的。

    样例输入

    tree.in 5 3 1 1 1 1 1 1 3 2 2 4 2 4 5 1 tree.out 3 tree.in 4 3 3 2 4 1 1 2 4 2 3 5 2 4 6 tree.out 8

    样例输出

    【输入输出样例解释1】 如图是满足样例的其中一种方案,最少要种3棵树。 【输入输出样例解释2】 如图是满足样例的其中两种方案,左图的方案需要种9棵树,右图的方案需要种8棵树。可以验证,最少需要种8棵树。

    提示

    【数据范围】


    对于30%的数据,0<n≤100,0<m≤100,ki=1;


    对于50%的数据,0<n≤2,000,0<m≤5,000,0<ki≤100;


    对于70%的数据,0<n≤50,000,0<m≤100,000,0<ki≤1,000;


    对于100%的数据,0<n≤500,000,0<m≤500,000,0<ki≤5,000

    这道题据说是差分约束路径的模板题
     令Si表示前i个村庄的种树的和,题目给出的条件可以化为差分形式:
       case 1:在第i个到第j个村庄至少中k棵树<==>  S[j]-S[i-1]>=k
               所以s[i-1]<=s[j]-k 即j到i-1有一条为-k的边
       case 2:第i个村庄最多种l棵树<==> S[i]-S[i-1]<=l
        所以S[i]<=S[i-1]+1 即i-1到i有一条长为l的边
       case 3:每个村庄至少种0棵树<==> S[i]>=S[i-1]
        所以S[i-1]<=S[i]+0,即i到i-1有一条长为0的边
     按上述规则加边,一边最短路即可
     输出dist[n]-dist[0]l
      小技巧:可以只加一部分边,其余每次松弛时判断即可
      
    #include<iostream>
    #include<cstdio>
    #include<algorithm>
    #include<cstring>
    using namespace std;
    #define INF 0x7FFFFFFF
    #define N 500005
    #define M 2000000
    int dist[N],num=0,ans=0,a[N];
    int n,m;
    struct note{
        int u,v,c;
    }Edge[M];
    void debug()
    {
        for(int i=0;i<=n;i++)
        cout<<"i="<<i<<" dist="<<dist[i]<<endl;
        cout<<"***********************
    "; 
    }
    void make_way(int u,int v,int z)
    {
        Edge[++num].u=u,Edge[num].v=v,Edge[num].c=z;
    }
    void bellman_ford(){
        int f=1;  int t;
        while(f){
            f=0;
            for(int i=1;i<=m;i++)   if(dist[Edge[i].v]>(t=dist[Edge[i].u]+Edge[i].c))  dist[Edge[i].v]=t,f=1;
            for(int i=1;i<=n;i++)   if(dist[i]-dist[i-1]>a[i])  dist[i]=dist[i-1]+a[i],f=1;
            for(int i=n;i;i--)      if(dist[i-1]>dist[i]) dist[i-1]=dist[i],f=1;
        }
    }
    int main()
    {
        scanf("%d %d",&n,&m);
        for(int i=1;i<=n;i++)
        {
            scanf("%d",&a[i]);
            //make_way(i-1,i,dist[i]),make_way(i,i-1,0);
        }
        for(int i=1;i<=m;i++)
        {
            int x,y,z;
            scanf("%d %d %d",&x,&y,&z);
            make_way(y,x-1,-z);
        }
        //debug();
        
        memset(dist,0,sizeof(dist));
        /*for(int i=1;i<=num;i++)
            printf("num:%d u:%d v:%d c:%d
    ",i,Edge[i].u,Edge[i].v,Edge[i].c);
            cout<<-dist[0]<<endl;*/
        bellman_ford();    
        cout<<dist[n]-dist[0]<<endl;
         
    } 
      
         
         
     
  • 相关阅读:
    小球(总结sort和cmp函数、结构体排序)
    垃圾装袋(标记法)【标记思想】
    种树(标记思想)【贪心算法】
    PHP 配置文件
    最大前驱路径
    PHP代码片段
    PHP 中的Trait
    BootStrapTable 错误
    工作两周总结
    工作一周总结
  • 原文地址:https://www.cnblogs.com/dancer16/p/7017945.html
Copyright © 2011-2022 走看看