CF-311B Cats Transport(斜率优化DP)

题目链接
题目描述
小S是农场主,他养了 \(M\)只猫,雇了 \(P\) 位饲养员。
农场中有一条笔直的路,路边有 \(N\) 座山,从 \(1\)\(N\)编号。
\(i\) 座山与第 \(i-1\) 座山之间的距离为 \(D_i\)
饲养员都住在 \(1\) 号山。
有一天,猫出去玩。
\(i\) 只猫去 \(H_i\)号山玩,玩到时刻 \(T_i\)
停止,然后在原地等饲养员来接。
饲养员们必须回收所有的猫。
每个饲养员沿着路从 $1 $号山走到 N 号山,把各座山上已经在等待的猫全部接走。
饲养员在路上行走需要时间,速度为\(1\)米/单位时间。
饲养员在每座山上接猫的时间可以忽略,可以携带的猫的数量为无穷大。
例如有两座相距为 1 的山,一只猫在 2 号山玩,玩到时刻 3 开始等待。
如果饲养员从 1 号山在时刻 2 或 3 出发,那么他可以接到猫,猫的等待时间为 0 或 1。
而如果他于时刻 1 出发,那么他将于时刻 2 经过 2 号山,不能接到当时仍在玩的猫。
你的任务是规划每个饲养员从 1 号山出发的时间,使得所有猫等待时间的总和尽量小。
饲养员出发的时间可以为负。

分析
接猫是任务,p个饲养员,每个饲养员接猫可以看作把几个猫放到一个集合。
\(i\)个猫被一个饲养员从1号点出发去接,等待时间与饲养员出发时刻有关。但出发时刻必须大于\(T[i] -\sum_{1}^iD[i]\)。将这个时间排个序,可以把这个猫看作若干个任务,可以贪心的证明把这些排序后的任务分成若干个不相交的部分会是最优的,如果相交了会有多余的花费(脑部一下,中间空出来的几个分到别的组,这几个猫的等待时间白白增加)。
假设算出了前\(k-1\)个饲养员的所有解。\(d[k][i]\)表示前\(k\)个饲养员接走前 \(i\)只猫时的答案。转移方程呼之欲出
\[d[k][i] = max\{d[k-1][j] + A_i*(i-j)-(s_i-s_)\},s[i] = \sum_1^iA[i]\]
把max去掉,得到最优的\(j\)满足
\[d[k-1][j]+s_j = A_i*j+d[k][i]-A_i*i+s_i\]
标准斜率优化DP,\(A_i\)递增。
另外由于P最大200,所以可以滚动数组优化掉一维

#include <bits/stdc++.h>
using namespace std;
const int N = 1e5+10;
typedef long long ll;
ll H[N],D[N],A[N],s[N],d[2][N],n,m,p,T[N];
int q[N];
int main(){
    scanf("%lld%lld%lld",&n,&m,&p);
    for(int i=2;i<=n;i++){
        scanf("%lld",&D[i]);
        D[i] += D[i-1];
    }
    for(int i=1;i<=m;i++){
        scanf("%lld%lld",&H[i],&T[i]);
        A[i] = T[i] - D[H[i]];
    }
    sort(A+1,A+1+m);
    for(int i=1;i<=m;i++){
        s[i] = s[i-1] + A[i];
    }
    for(int i=1;i<=m;i++)d[1][i] = 1ll * i * A[i] - s[i];
    for(int k=2,w=0;k<=p;k++,w^=1){
        int l = 0,r = 0;
        for(int i=1;i<=m;i++){
            while(l < r && (d[w^1][q[l+1]] - d[w^1][q[l]] + s[q[l+1]] - s[q[l]]) <= A[i] * ((q[l+1] - q[l])))l++;
            int j = q[l];
            //cout << l << ' ' << r << ' ' << j << ' ' << d[w^1][j] << endl;
            d[w][i] = d[w^1][j] + 1ll * (i-j) * A[i] - (s[i]-s[j]);
            while(l < r && (d[w^1][q[r-1]] - d[w^1][q[r]] + s[q[r-1]] - s[q[r]]) * (q[r-1]-i) > (d[w^1][q[r-1]] - d[w^1][i] + s[q[r-1]] - s[i]) * (q[r-1] - q[r]))r--;
            q[++r] = i;
        }
    }
    printf("%lld\n",d[p&1][m]);
    return 0;
}

转载于:https://www.cnblogs.com/1625--H/p/11276564.html

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
map pfn expected mapping type uncached-minus for [mem 0x7c11f000-0x7c11ffff], got write-back 这个问题怎么解决,这个会导致系统死机吗?PID: 500 TASK: ffff8800740d6dd0 CPU: 4 COMMAND: "mate-settings-d" #0 [ffff88024a6e7988] machine_kexec at ffffffff81059cdb #1 [ffff88024a6e79e8] __crash_kexec at ffffffff81105182 #2 [ffff88024a6e7ab8] crash_kexec at ffffffff81105270 #3 [ffff88024a6e7ad0] oops_end at ffffffff8168ed88 #4 [ffff88024a6e7af8] no_context at ffffffff8167e993 #5 [ffff88024a6e7b48] __bad_area_nosemaphore at ffffffff8167ea29 #6 [ffff88024a6e7b90] bad_area_nosemaphore at ffffffff8167eb93 #7 [ffff88024a6e7ba0] __do_page_fault at ffffffff81691b1e #8 [ffff88024a6e7c00] do_page_fault at ffffffff81691cc5 #9 [ffff88024a6e7c30] page_fault at ffffffff8168df88 [exception RIP: dev_set_drvdata+26] RIP: ffffffff8142c60a RSP: ffff88024a6e7ce8 RFLAGS: 00010206 RAX: 0000000900000000 RBX: ffff880258686098 RCX: 0000000180040001 RDX: ffff8801849e4000 RSI: 0000000000000000 RDI: ffff880258686098 RBP: ffff88024a6e7cf8 R8: ffff8801849e4000 R9: 0000000180040001 R10: 00000000849e6001 R11: ffffea0006127800 R12: ffff880239383398 R13: ffff880239383300 R14: ffff880061c29d08 R15: 0000000000000246 ORIG_RAX: ffffffffffffffff CS: 0010 SS: 0018 #10 [ffff88024a6e7d00] snd_usb_audio_free at ffffffffa059a587 [snd_usb_audio] #11 [ffff88024a6e7d28] snd_usb_audio_dev_free at ffffffffa059a5b2 [snd_usb_audio] #12 [ffff88024a6e7d38] __snd_device_free at ffffffffa02e2dc9 [snd] #13 [ffff88024a6e7d50] snd_device_free_all at ffffffffa02e311b [snd] #14 [ffff88024a6e7d70] release_card_device at ffffffffa02dd7de [snd] #15 [ffff88024a6e7d90] device_release at ffffffff814273f2 #16 [ffff88024a6e7db8] kobject_release at ffffffff8131a29e #17 [ffff88024a6e7de8] kobject_put at ffffffff8131a158 #18 [ffff88024a6e7e00] put_device at ffffffff81427717 #19 [ffff88024a6e7e10] snd_card_file_remove at ffffffffa02de1b4 [snd] #20 [ffff88024a6e7e40] snd_ctl_release at ffffffffa02df421 [snd] #21 [ffff88024a6e7e78] snd_disconnect_release at ffffffffa02ddafd [snd] #22 [ffff88024a6e7ea8] __fput at ffffffff811fff09 #23 [ffff88024a6e7ef0] ____fput at ffffffff812001be #24 [ffff88024a6e7f00] task_work_run at ffffffff810accc7 #25 [ffff88024a6e7f30] do_notify_resume at ffffffff8102ab22 #26 [ffff88024a6e7f50] int_signal at ffffffff8169677d gdb调试的bt,这个怎么判断是什么造成的死机
07-20

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值