POJ1821 Fence 单调队列优化DP
http://poj.org/problem?id=1821 题意:给长度为n的木板,k个工人,每个工人要么不粉刷,或者选择一个包含木板si,长度不超过li的连续的一段木板粉刷,每粉刷一块得到pi的报酬,问如何安排工人使得总报酬最大? 思路:可以按si给工人排序,这样我们就可以按照顺序依次安排工人。设f[i][j]表示到第i个工人,刷到前j块木板的最大报酬, 三种情况 工人不刷:f[i][j]=f[i-1][j] 木板空着:f[i][j]=f[i][j-1] 第i个工人刷k+1到j的木板,其中k,j满足k+1<=si<=j,报酬就是f[i][j]=f[i-1][k]+pi*(j-k),拆开得pi*j+(-pi*k+f[i-1][k])。可以发现对于每个j来说,要找到它对应的最大报酬,需要尝试范围内的所有k值,而实际上我们可以发现随着枚举j,j的增大,所对应的k的区间范围的上下界也是递增的,这样的话,我们就可以考虑使用单调队列来优化枚举k的过程,又因为要使报酬最大,所以我们需要维护一个单调递减的序列来维护-pi*k+f[i-1][k]这个只与k相关的最大值。 #include<cstdio> #include<algorithm> using namespace std; const int maxn=16005; struct note { int l,p,s; } a[maxn]