关于二进制优化这一点,它为什么正确,为什么合理,凭什么可以这样分,至少我是花了很久很久才理解的,先拿一道题来说吧。
HDU 2844 Coins
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2844
题目:
3 10 1 2 4 2 1 1 2 5 1 4 2 1 0 0
8 4
#include
#include
#include
#include
#include
#define maxn 100005
using namespace std;
int dp[maxn],a[105],c;
int main(){
int n,m;
while(cin>>n>>m){
if(n==0 || m==0) break;
memset(dp,0,sizeof(dp));
for(int i=1;i<=n;i++)
scanf("%d",&a[i]);
for(int i=1;i<=n;i++){ //物品件数遍历
scanf("%d",&c);
if(m=容量
for(int k=0;k<=m;k++) //完全背包
if(k>=a[i])
dp[k]=max(dp[k],dp[k-a[i]]+a[i]);
}
else{
for(int j=1;j<=c;j<<=1){ //分堆过程
for(int k=m;k>=0;k--) //一次性询问2^j的01背包
if(k>=a[i]*j)
dp[k]=max(dp[k],dp[k-a[i]*j]+a[i]*j);
c-=j;
}
if(c>0){ //小优化所在,给你一个眼神儿自己体会
for(int k=m;k>=0;k--)
if(k>=a[i]*c)
dp[k]=max(dp[k],dp[k-a[i]*c]+a[i]*c);
}
}
}
int ans=0;
for(int i=1;i<=m;i++){
if(dp[i]==i)
ans++;
}
printf("%d\n",ans);
}
return 0;
}
我们首先确认三点:
(1)我们知道转化成01背包的基本思路就是判断我是取了你好呢还是不取你好。
(2)我们知道任意一个实数可以由二进制数来表示,也就是2^0~2^k其中一项或几项的和。
(3)这里多重背包问的就是每件物品取多少件可以获得最大价值。
三者综合,然后在纸上深究一下,大概就能理解了。
如果直接遍历转化为01背包问题,是每次都拿一个来问,取了好还是不取好,假如10个取7个好,那么在实际的遍历过程中在第七个以后经过状态转移方程其实已经是选择“不取”好了,现在,用二进制思想将其分堆,分成k+1个分别有2^k个的堆,然后拿这一堆一堆去问,我是取了好呢,还是不取好呢,经过dp选择之后,结果和拿一个一个来问的结果是完全一样的,因为dp选择的是最优结果,而根据第二点任意一个实数都可以用二进制来表示,如果最终选出来10个取7个是最优的在分堆的选择过程中分成了2^0=1,2^1=2,2^2=4,2^3=8这四堆,然后去问四次,也就是拿去走dp状态转移方程,走的结果是第一堆1个,取了比不取好,第二堆2个,取了比不取好,第三堆四个,取了比不取好,第四堆8个,取了还不如不取,最后依旧是取了1+2+4=7个,为什么是这样呢?因为dp本身就是用来比较哪个更优的,在状态转移的过程中自然会完成上述询问得出相应结果。这也就是说,无论最终取几个是最优解,用二进制取出来的结果和一次一次问是完全一样的。相信已经说的足够详细清楚了。重要还是自己体会一下为什么这样。
理解之后看代码发现事实上在写代码的过程中又搞了一丢丢小优化,自己感觉一下吧~
有没有被二进制惊艳到?就这样复杂度顿时降了一个档次,瞬间感觉世界真奇妙~~~神奇的二进制~~~ok,打字打的好累,休息一会儿继续写代码~