zoukankan      html  css  js  c++  java
  • Cats Transport

    CF

    洛咕

    Vjudge

    题意:小S是一个大农场主.他养了m只可爱的猫,雇佣了p个饲养员.农场中有一条笔直的路,路边有N座山,从1到N编号.山丘i与山丘i-1的距离是(d_i)米.饲养员都住在1号山丘.一天,猫外出玩耍.猫i去山丘(h_i)游玩,在(t_i)时间结束他的游玩,然后在原地傻等饲养员.饲养员必须把所有的猫带回.每个饲养员直接从1号山走到N号山,不花费时间的把已经游玩结束的猫带回.每个铲屎官的速度为一米每单位时间,并且足够强壮来带上任意数量的猫.举个栗子,假装我们有两个相距为1的山丘,有一只猫,他想去山丘2玩到时间3.然后饲养员如果在时间2或者时间3从1号山丘出发,他就能抱走猫.如果他在时间1出发那么就不行(猫子还在玩耍).如果饲养员在时间2出发,猫就不用等他(ΔT=0).如果他在时间3出发,猫就要等他1个单位时间。你的任务是安排每个饲养员出发的时间,最小化所有猫等待的时间之和.

    分析:对于每只猫,设(A[i]=t[i]-sum_{j=1}^{h[i]}d[j]),一名饲养员如果想接到第i只猫,就必须要在(A[i])时刻之后从1号山出发.若出发时刻为t,则猫的等待时间就是(t-A[i])

    预处理出A数组,并从小到大排序,这样一个饲养员接走的猫一定是连续的一群,这样本题就是斜率优化的模板题了.

    (f[i][j])表示前i个饲养员接走前j只猫的最小等待时间,则有,

    (f[i][j]=f[i-1][k]+A[j]*(j-k)-(S[j]-S[k]))其中S数组是A数组排好序后的前缀和数组

    设l<k且k比l更优,即

    (f[i-1][k]+A[j]*(j-k)-(S[j]-S[k])<f[i-1][l]+A[j]*(j-l)-(S[j]-S[l]))

    整理一下上式得到,

    (frac{f[i-1][k]+S[k]-f[i-1][l]-S[l]}{k-l}<A[j])

    //#include<bits/stdc++.h>
    #include<iostream>
    #include<cstdio>
    #include<cstring>
    #include<algorithm>
    #define LL long long
    using namespace std;
    inline int read(){
       int s=0,w=1;char ch=getchar();
       while(ch<'0'||ch>'9'){if(ch=='-')w=-1;ch=getchar();}
       while(ch>='0'&&ch<='9'){s=s*10+ch-'0';ch=getchar();}
       return s*w;
    }
    const int N=200005;
    const int M=100005;
    int d[N],h[M],t[M],q[N];
    LL D[N],sum[M],a[M],f[105][M];
    int main(){
        int n=read(),m=read(),p=read();
        for(int i=2;i<=n;i++){
    		d[i]=read();
    		D[i]=D[i-1]+d[i];
        }
        for(int i=1;i<=m;i++){
    		h[i]=read();t[i]=read();
    		a[i]=t[i]-D[h[i]];
        }
        sort(a+1,a+m+1);
        for(int i=1;i<=m;i++)sum[i]=sum[i-1]+a[i];
        memset(f,0x3f,sizeof(f));
        for(int i=0;i<=p;i++)f[i][0]=0;
        for(int i=1;i<=p;i++){
    		int l=1,r=1;q[1]=0;
    		for(int j=0;j<=m;j++){
    	    	while(l<r&&(f[i-1][q[l+1]]+sum[q[l+1]])-(f[i-1][q[l]]+sum[q[l]])<=(q[l+1]-q[l])*a[j])l++;
    	    	f[i][j]=f[i-1][q[l]]+sum[q[l]]+a[j]*j-sum[j]-a[j]*q[l];
    	    	while(l<r&&((f[i-1][q[r]]+sum[q[r]])-(f[i-1][q[r-1]]+sum[q[r-1]]))*(j-q[r])>=((f[i-1][j]+sum[j])-(f[i-1][q[r]]+sum[q[r]]))*(q[r]-q[r-1]))r--;
    	    	q[++r]=j;
    		}
        }
        printf("%lld
    ",f[p][m]);
        return 0;
    }
    
    
  • 相关阅读:
    源代码搭建应用(一)——动手搭建自己的计算集群系统
    [转载]Ubuntu 14.04中root 密码忘记解决方法
    如何用路由器改成WiFi Pineapple系统镜像网络流量
    DELPHI黑客编程(一):正向后门原理实现
    OpenWrt 路由器过滤广告的N种方法
    Windows 10 上强制Visual Studio以管理员身份运行
    DB2修改表字段
    Git Fast-forward提交
    C#(去、过滤)掉字符中的换行符
    VS读取文件或写入文件时出现中文乱码问题
  • 原文地址:https://www.cnblogs.com/PPXppx/p/11007525.html
Copyright © 2011-2022 走看看