【POJ1276】Cash Machine(多重背包单调队列优化)


大神博客转载http://www.cppblog.com/MatoNo1/archive/2011/07/05/150231.aspx
多重背包的单调队列初中就知道了但一直没(不会)写
二进制优化初中就写过
一直不写会心虚就写一下这个吧
朴素方程
dp[i,j]=max(dp[i-1,j-w[i]*k]+c[i]*k) w[i]*k<=j k<=j div w[i]
忽略第一维
dp[j]=max(dp[j-w[i]*k]+c[i]*k)
复杂度O(m2n)
以下是大神原文:
将决策下标按照模w0的余数进行分类,可以分成w0类,分别对应模w0余0、余1……余(w0-1)的情况。这时,上面方程中的所有决策下标j-x*w0都是同一类的。进一步,设q =j/w0(下取整),r=j%w0,则j=q*w0+r,对于某个决策下标j',设k=(j'-r)/w0,即j'=k*w0+r。显然可以发现,k的取值范围是:k>=0且q-s0<=k<=q,也即k的下界是max{0, q-s0}——随j的单调而单调。
然后,转移方程可以改为(这里把r当成一个已知量了):
F[q*w0+r] = max{F[k*w0+r]+(q-k)*v0} (k>=0且q-s0<=k<=q)
即F[q*w0+r]=max{F[k*w0+r]-k*v0}+q*v0 (k>=0且q-s0<=k<=q)
设G[k]=F[k*w0+r]得:
G[q]=max{G[k]-k*v0}+q*v0 (k>=0且q-s0<=k<=q)
这个方程已经可以使用单调队列来优化了!

这样可以得出算法:
(1)从1到n,枚举i,建立w[i]个空的单调队列,每个队列的元素都是两个int值:(k, val),表示转换后下标和决策值(G[k]-k*v[i]);
(2)从0到m,枚举j,得出q、r的值,对于队列r:
【1】删去队首过时(k<q-m[i])的元素;
【2】F[j]入队(这里的F[j]指上一阶段的F[j],即F[i-1][j]。因此这一步操作一定要先进行),删去队尾所有决策值val不大于(F[j]-q*v[i])的元素。
【3】取出队首结点,其val值加上q*v[i]后即为本阶段F[j]的值。
最后F[m]即为结果。总时间复杂度为O(NM)。

其实这个是可以推广的,即对于如下形式的转移方程(其中H、G和W均为常量,B[i]为决策下标的下界,随i单调):
F[i] = opt{F[i-x*H+W]}+G (B[i]<=i-x*H+W<i,x∈N)
都可以用上述的办法进行转化,从而进行单调队列优化。

我自己的萎靡过程:
形如dp[i]=max(dp[j]+cost(j+1,i))+M,cost具有单调性的DP可以用单调队列优化
我们尝试转化它的形式
dp[j]=max(dp[j-w[i]*1]+c[i]*1,dp[j-w[i]*2]+c[i]*2])……
设k<j<i,i-j,i-k能整除w[i],显然j-k也可以
dp[j]+c[i]*(i-j)/w[i]>dp[k]+c[i]*(i-k)/w[i]
dp[j]-dp[k]>c[i]*(i-k-i+j)/w[i]
dp[j]-dp[k]>c[i]*(j-k)/w[i]
好像很麻烦的样子

好吧还是学大神的做法吧
设j=w[i]*k+r,r=j mod w[i]
dp[w[i]*k+r]=max(dp[w[i]*(k-x)+r]+c[i]*x)
设x>y,dp[w[i]*x+r]+c[i]*x>dp[w[i]*y+r]+c[i]*y
dp[w[i]*x+r]-dp[w[i]*y+r]>c[i]*(y-x)
(dp[w[i]*x+r]-dp[w[i]*y+r])/(y-x)>c[i]对于单个i成立
我们对于每个j计算出它对于每个i时的max x与r O(mn)
之后就可以用单调队列的思路做了

POJ上一直RE不知道怎么回事搞得我都不知道对不对

自己的对拍好像没有问题

哪位找到错误提醒下谢谢

一些奇奇怪怪的if 判 a[i] k 什么的 不加会有奇奇怪怪的错误

显然空间不够必须滚动队列

POJ空间也卡 格式也卡

懒得改了 反正没什么乱用

 1 var q:array[0..1000,0..1000,1..2]of longint;
 2     f:array[0..1,0..100000]of longint;
 3     t,w:array[0..20000]of longint;
 4     a,ww,cc:array[1..100]of longint;
 5     m,n,i,j,k,r,t1,w1,tmp,ans,v:longint;
 6 
 7 begin
 8  assign(input,'1.in'); reset(input);
 9  assign(output,'1.out'); rewrite(output);
10  while not eof do
11  begin
12   read(m,n);
13   if (m=0)and(n=0) then break;
14   for i:=1 to n do
15   begin
16    read(a[i],ww[i]);
17    cc[i]:=ww[i];
18    if a[i]>m div ww[i] then a[i]:=m div ww[i];
19   end;
20   fillchar(f,sizeof(f),0); ans:=0;
21   for i:=1 to n do
22   if a[i]>0 then
23   begin
24    v:=1-v;
25    fillchar(q,sizeof(q),0);
26    for j:=0 to m do f[v,j]:=f[1-v,j];
27    for j:=0 to ww[i]-1 do begin w[j]:=1; t[j]:=1; end;
28    for j:=0 to m do
29    begin
30     k:=j div ww[i];
31     if k>a[i] then continue; r:=j mod ww[i];
32     inc(w[r]); q[r,w[r] mod 1000,1]:=k; q[r,w[r] mod 1000,2]:=f[1-v,j]-k*ww[i];
33    end;
34 
35    for j:=0 to m do
36    begin
37     k:=j div ww[i]; r:=j mod ww[i];
38     t1:=t[r]; w1:=w[r];
39     while (w1-t1>=1)and(k-q[r,t1 mod 1000,1]>a[i]) do
40     begin
41      inc(t[r]); inc(t1);
42     end;
43 
44      tmp:=q[r,t1 mod 1000,2]+cc[i]*k;
45     if tmp>ans then ans:=tmp;
46     if tmp>f[v,j] then f[v,j]:=tmp;
47 
48 
49     while (w1-t1>=0)and(q[r,w1 mod 1000,2]<=f[1-v,j]-k*ww[i]) do
50      begin dec(w[r]); dec(w1); end;
51 
52     
53      inc(w[r]); inc(w1); q[r,w1 mod 1000,1]:=k; q[r,w1 mod 1000,2]:=f[1-v,j]-k*ww[i];
54  
55    end;
56   end;
57   writeln(ans);
58 
59  end;
60  close(input);
61  close(output);
62 end.
View Code
原文地址:https://www.cnblogs.com/myx12345/p/5528502.html