码迷,mamicode.com
首页 > 其他好文 > 详细

斜率dp的模板总结

时间:2019-08-11 10:32:19      阅读:87      评论:0      收藏:0      [点我收藏+]

标签:pen   pre   ble   总结   amp   ++   ==   str   斜率dp   

#include<cstdio>
#include<algorithm>
using namespace std;
long long sumt[40005],sum[40005],f[40005],q[40005];
double X(long long x)
{
    return sumt[x];
} 
double Y(long long x)
{
    return (f[x]+sum[x]);
}
double slope(long long a,long long b)
{
    return ((Y(a)-Y(b))/(X(a)-X(b)));
}
struct ben
{
    long long t,r;
}a[40005],b[40005];
long long cmp(const ben &a,const ben &b)
{
    return a.t>b.t;
}
int main()
{
    freopen("nt2011_design.in","r",stdin);
    freopen("nt2011_design.out","w",stdout); 
    long long l=0,r=0;
    long long n,m;
    scanf("%lld%lld",&n,&m);
    for(long long i=1;i<=n;i++)
    {
        scanf("%lld%lld",&a[i].t,&a[i].r);
    }
    sort(a+1,a+n+1,cmp);
    for(long long i=1;i<=n;i++)
    {
        if(a[i].t==a[i+1].t)
        {
            a[i+1].r+=a[i].r;
            a[i].r=0;
        }
    }
    long long cnt=0;
    for(long long i=1;i<=n;i++)
    {
        if(a[i].r!=0)
        {
            b[++cnt]=a[i];
        }
    }
    long long maxt=b[1].t;
    b[cnt+1].t=0;
    b[cnt+1].r=0;
    cnt++;
    for(long long i=1;i<=cnt;i++)
    {
        b[i].t=maxt-b[i].t;
        sumt[i]=sumt[i-1]+b[i].r;
        sum[i]=sum[i-1]+b[i].t*b[i].r;
    }
    for(long long i=1;i<=cnt;i++)
    {
        while(l<r&&slope(q[l+1],q[l])<b[i].t)l++;
        f[i]=f[q[l]]+b[i].t*(sumt[i]-sumt[q[l]])-(sum[i]-sum[q[l]]);
        if(i!=cnt)f[i]+=m;
        while(l<r&&slope(i,q[r-1])<slope(q[r-1],q[r]))r--;
        q[++r]=i;
    }
    printf("%lld\n",f[cnt]);
    return 0; 
}

斜率dp的模板总结

标签:pen   pre   ble   总结   amp   ++   ==   str   斜率dp   

原文地址:https://www.cnblogs.com/ShineEternal/p/11333869.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!