[DP之多重背包优化方法]
首先我们看一道有趣的题目

然后这道题很快想到是一个多重背包和无限背包混合体
那么我们就以这道题 来讨论一下多重背包的优化
首先我们看看朴素打法
memset(F,,sizeof(F)); F[]=;
for(int i=;i<=N;i++)
for(int k=;k<=C[i];k++)
for(int j=T*;j>=;j--)
if(j-V[i]>=) F[j]=min(F[j],F[j-V[i]]+);
很简单 很好懂 但是这样做导致时间复杂度为O(N*C*T) 这道题来看超时到爆炸
那么我们考虑两种方法 第一种是像wph写的 首先贪心一会儿 然后再背包 答案大概在T到2*T的范围内
但是这种方法导致代码很玄学 因为贪心的时候很有可能有一些东西处理不到
那么我们考虑第二种方法 就是优化多重背包 因为无限背包的时间是足够的
优化多重背包有两种算法,但是下面只介绍最快的:
1.用二进制优化 这个好像我有写过 但不知道怎么打和原理了 陈年老事 具体可以问wph 时间复杂度O(NTlog(C))
2.可以用单调队列进行优化 时间复杂度为O(NT)
我只讨论单调队列优化 我们先列出有
F[i][j]=min(F[i-1][j-vi*k]+k) (0<=k<=ci)
那么我们发现 每一次都是和vi有关的 我们其实可以把式子变一下
令a=j/v[i] b=j%v[i] j=a*v[i]+b
F[j]=min(F[b+k*v[i]]-k)+a (a-c[i]<=k<=a)
我们把k枚举 就是单调队列了,具体看下面
其实我们在做j这个状态的时候 k和a是相同的 当然方便理解 因为以前的是k 现在的是a 只是对于当前来说我的a和k是一样 但是对于别的来说 我就要找最小的k
我们可以枚举b 然后可以每次都 +v[i] 处理 这样的话每一次每个位置只会被扫1遍 大大减少了时间 所以时间复杂度为O(NT)很显然
memset(F,,sizeof(F)); F[]=;
for(int i=;i<=N;i++)
{
for(int j=;j<=;j++) G[j]=F[j]; for(int b=;b<V[i];b++)
{
head=; tail=;
for(int j=b;j<=;j+=V[i])
{
int a=j/V[i];
while(head<=tail&&Q[head].first<a-C[i]) head++;
if(head<=tail) F[j]=min(G[j],Q[head].second+a);
while(head<=tail&&Q[tail].second>=G[j]-a) tail--;
Q[++tail]=make_pair(a,G[j]-a); }
}
}
良心的贴一道这道例题的链接 这里
[DP之多重背包优化方法]的更多相关文章
- 单调队列优化DP,多重背包
单调队列优化DP:http://www.cnblogs.com/ka200812/archive/2012/07/11/2585950.html 单调队列优化多重背包:http://blog.csdn ...
- dp之多重背包(二进制优化)
void solve(int v,int w,int c){ int count=0; for(int k=1;k<=c;k<<=1) { val[c ...
- 硬币问题 (dp,多重背包的二分优化)
题目描述 给你n种硬币,知道每种的面值Ai和每种的数量Ci.问能凑出多少种不大于m的面值. 输入 有多组数据,每一组第一行有两个整数 n(1≤n≤100)和m(m≤100000),第二行有2n个整数, ...
- hdu1059&poj1014 Dividing (dp,多重背包的二分优化)
Problem Description Marsha and Bill own a collection of marbles. They want to split the collection a ...
- HDU 3732 Ahui Writes Word 多重背包优化01背包
题目大意:有n个单词,m的耐心,每个单词有一定的价值,以及学习这个单词所消耗的耐心,耐心消耗完则,无法学习.问能学到的单词的最大价值为多少. 题目思路:很明显的01背包,但如果按常规的方法解决时间复杂 ...
- dp之多重背包hdu1059
题意:价值为1,2,3,4,5,6. 分别有n[1],n[2],n[3],n[4],n[5],n[6]个.求能否找到满足价值刚好是所有的一半的方案. 思路:简单的多重背包,我建议多重背包都用二进制拆分 ...
- hdu1059(多重背包优化)
使用一种二进制的优化, 可以完美的解决这题,<背包九讲>中说的非常好 但是还有一种线性复杂的算法. 应该算是该题很巧妙的解法 ;i++) { ;l--) { ) continue; ;k& ...
- nyoj 546——Divideing Jewels——————【dp、多重背包板子题】
Divideing Jewels 时间限制:1000 ms | 内存限制:65535 KB 难度:4 描述 Mary and Rose own a collection of jewells. ...
- hdu 1059 Dividing(多重背包优化)
Dividing Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total Su ...
随机推荐
- KMP算法简单回顾
前言 虽从事企业应用的设计与开发,闲暇之时,还是偶尔涉猎数学和算法的东西,本篇根据个人角度来写一点关于KMP串匹配的东西,一方面向伟人致敬,另一方面也是练练手,头脑风暴.我在自娱自乐,路过的朋友别太认 ...
- 30个HTML初学者建议
The most difficult aspect of running Nettuts+ is accounting for so many different skill levels. If w ...
- rapid-framework脚手架快速搭建springMVC框架项目
rapid-framework介绍: 一个类似ruby on rails的java web快速开发脚手架,本着不重复发明轮子的原则,框架只是将零散的struts(struts2)+spring+h ...
- IOC 容器在 ASP.NET MVC 中的应用
IOC 容器在 ASP.NET MVC 中的应用 IOC:Inversion Of Control 翻译为控制反转,我们在面向对象软件开发过程中,一个应用程序它的底层结构可能由N种不同的构件来相互协作 ...
- win7 64位系统装oracle11 提示环境变量path 值超过1023字符
win7 64位系统装oracle10 提示环境变量path 值超过1023字符 1.提示环境变量path 值超过1023字符. 方案: 1.oracle安装文件,右键属性--设置兼容性--兼容XP ...
- 把《C语言接口与实现》读薄之第一章:引言
1.1文学程序 文学程序(literate program):接口及其实现的代码与对其进行解释的正文交织在一起.文学程序由英文正文和带标签的程序代码块组成.例如, 〈compute x * y〉≡ s ...
- 在ASP.NET MVC3 中利用Jsonp跨域访问
在ASP.NET MVC3 中利用Jsonp跨域访问 在信息系统开发的时,根据相关业务逻辑难免会多系统之间互相登录.一般情况下我们需要在多系统之间使用多个用户名和密码.这样客户就需要在多个系统之间重复 ...
- GNU Make 学习系列一:怎样写一个简单的Makefile
编程通常遵循一个相当简单的程序:编辑源文件,编译源代码成可执行的格式,调试结果.尽管将源代码翻译成可执行程序是常规的过程,如果做的不正确,程序员可能会浪费大量的时间去追踪问题.大多数的开发者都经历过这 ...
- 个人计算机安装hadoop全分布
一.工具说明 设备:实体主机一台 校园网络 虚拟设备:VMware下安装的ubuntu12.04版本三台,分别是master,slave1,slave2 二.ubuntu安装篇 前提知识:在windo ...
- CSS盒子的浮动
web前端学习笔记(CSS盒子的浮动) 在标准流中,一个块级元素在水平方向会自动伸展,直到包含它的元素的边界:而在竖直方向和兄弟元素依次排列,不能并排.使用“浮动”方式后,块级元素的表现就会有所不同. ...