0/1背包问题的动态规划法
背包问题:
在M件物品取出若干件放在空间为W的背包里,每件物品的重量为W1,W·2……Wn,与之相对应的价值为P1,P2……Pn。求出获得最大价值的方案。
注意:在本题中,所有的重量值均为整数。
[算法分析]:
对于背包问题,通常的处理方法是搜索。
用递归来完成搜索,算法设计如下:
function make( i {处理到第i件物品} , j{剩余的空间为j}:integer) :integer;
初始时i=m , j=背包总容量
begin
if i:=0 then
make:=0;
if j>=wi then (背包剩余空间可以放下物品 i )
r1:=make(i-1,j-wi)+v[i]; (第i件物品放入所能得到的价值 )
r2:=make(i-1,j) (第i件物品不放所能得到的价值 )
make:=max{r1,r2}
end;
这个算法的时间复杂度是o(2^n),我们可以做一些简单的优化。
由于本题中的所有物品的重量均为整数,经过几次的选择后背包的剩余空间可能会相等,在搜索中会重复计算这些结点,所以,如果我们把搜索过程中计算过的结点的值记录下来,以保证不重复计算的话,速度就会提高很多。这是简单的"以空间换时间"。
我们发现,由于这些计算过程中会出现重叠的结点,符合动态规划中子问题重叠的性质。
同时,可以看出如果通过第n次选择得到的是一个最优解的话,那么第n-1次选择的结果一定也是一个最优解。这符合动态规划中最优子问题的性质。
考虑用动态规划的方法来解决,这里的:
阶段是:在前n件物品中,选取若干件物品放入背包中;
状态是:在前n件物品中,选取若干件物品放入所剩空间为w的背包中的所能获得的最大价值;
决策是:第n件物品放或者不放;
由此可以写出动态转移方程:
我们用f[i,j]表示在前 i 件物品中选择若干件放在所剩空间为 j 的背包里所能获得的最大价值
f[i,j]=max{f[i-1,j-wi]+pi (j>=wi), f[i-1,j]}
这样,我们可以自底向上地得出在前m件物品中取出若干件放进背包能获得的最大价值,也就是f[m,w]
算法设计如下:
procedure make;
begin
for i:=0 to w do
f[0,i]:=0;
for i:=1 to m do
for j:=0 to w do begin
f[i,j]:=f[i-1,j];
if (j>=w[i]) and (f[i-1,j-w[i]]+v[i]>f[i,j]) then
f[i,j]:=f[i-1,j-w[i]]+v[i];
end;
writeln(f[m,wt]);
end;
由于是用了一个二重循环,这个算法的时间复杂度是o(n*w)。而用搜索的时候,当出现最坏的情况,也就是所有的结点都没有重叠,那么它的时间复杂度是o(2^n)。看上去前者要快很多。但是,可以发现在搜索中计算过的结点在动态规划中也全都要计算,而且这里算得更多(有一些在最后没有派上用场的结点我们也必须计算),在这一点上好像是矛盾的。
事实上,由于我们定下的前提是:所有的结点都没有重叠。也就是说,任意n件物品的重量相加都不能相等,而所有物品的重量又都是整数,那末这个时候w的最小值是:1+2+2^2+2^3+……+2^n-1=2^n -1
此时n*w>2^n,动态规划比搜索还要慢~~|||||||所以,其实背包的总容量w和重叠的结点的个数是有关的。
考虑能不能不计算那些多余的结点……
那么换一种状态的表示方式:
状态是:在前n件物品中,选取若干件物品放入所占空间为w的背包中的所能获得的最大价值;
阶段和决策:同上;
状态转移方程是:
f[i,j]=max{f[i-1,j-wi]+pi (j+wi<=背包总容量), f[i-1,j]}
这样,我们可以得出在前m件物品中取出若干件放进背包在所占空间不同的状态下能获得的最大价值,在其中搜索出最大的一个就是题目要求的解。
算法设计如下:
procedure make;
begin
f[0,wt]:=0;
for i:=1 to n do
for j:=0 to w (背包总容量) do
if f[i-1,j]未被赋过值 then (这些结点与计算无关,忽略)
continue
else
f[i,j]:=max{f[i-1,j+wi]+pi , f[i-1,j]};
最大价值:=max{f[n,j]} (求最大值)
j:=1 to w
end;
由于事实上在计算的过程中每一个阶段的状态都只和上一个阶段有关,所以只需要来一个两层的数组循环使用就可以了,这是动态规划中较常使用的降低空间复杂度的方法。
本题能够用动态规划的一个重要条件就是:所有的重量值均为整数 因为
1)这样我们才可以用数组的形式来储存状态;
2)这样出现子问题重叠的概率才比较大。(如果重量是实型的话,几个重量相加起来相等的概率会大大降低)
所以,当重量不是整数的时候本题不适合用动态规划。
[解的输出]:
在计算最大价值的时候我们得到了一张表格(f[i,j]),我们可以利用这张表格输出解。
可以知道,如果f[i-1,j+wi]+v[i]=f[i,j] (第二个算法),则选择物品i放入背包。
算法设计1:
进行反复的递归搜索,依次输出物品序号;
procedure out(i,j:integer);(初始时 i=n, j=获得最大价值的物品所占的空间)
begin
if i=0 then exit;
if f[i,j]=f[i-1,j+w[i]]+v[i] then begin
输出解
out(i-1,j+w[i]);
end
else
out(i-1,j);
end;
算法设计2:
同样的思路我们可以用循环来完成;
procedure out2;
var
i,ws:integer;
begin
ws:=获得最大价值的物品所占的空间;
for i:=n downto 1 do begin
if (ws>=w[i]) and (f[i,ws]=f[i-1,ws-w[i]]+v[i]) then begin
输出解;
ws:=ws-w[i];
end;
end;
writeln;
end;
用这两种算法的前提是我们必须存住 f[i,j] 这一整个二维数组,但是如果用循环数组的话怎样输出解呢?
显然,我们只需要存住一个布尔型的二维数组,记录每件物品在不同的状态下放或者不放就可以了。这样一来数组所占的空间就会大大降低。
[解题收获]:
1)在动态程序设计中,状态的表示是相当重要的,选择正确的状态表示方法会直接影响程序的效率。
2)针对题目的不同特点应该选择不同的解题策略,往往能够达到事半功倍的效果。像本题就应该把握住"所有的重量值均为整数"这个特点。
--------------------------------------------------------------
背包问题全攻略
背包问题全攻略
部分背包问题可有贪心法求解:计算PiWi
数据结构:
w[i]第i个背包的重量;
p[i]第i个背包的价值;
(1)每个背包只能使用一次或有限次(可转化为一次):
A.求最多可放入的重量。
NOIP2001 装箱问题
有一个箱子容量为v(正整数,o≤v≤20000),同时有n个物品(o≤n≤30),
每个物品有一个体积 (正整数)。
要求从 n 个物品中,任取若千个装入箱内,使箱子的剩余空间为最小。
l 搜索方法
procedure search(k,vinteger); {搜索第k个物品,剩余空间为v}
var i,jinteger;
begin
if v best then best=v;
if v-(s[n]-s[k-1]) =best then exit;
{s[n]为前n个物品的重量和}
if k =n then begin
if v w[k] then search(k+1,v-w[k]);
search(k+1,v);
end;
end;
l DP
F[I,j]为前i个物品中选择若干个放入使其体积正好为j的标志,为布尔型。
实现将最优化问题转化为判定性问题
F[I,j]=f[i-1,j-w[i]] (w[I] =j =v) 边界:f[0,0]=true.
For I=1 to n do
For j=w[I] to v do F[I,j]=f[I-1,j-w[I]];
优化:当前状态只与前一阶段状态有关,可降至一维。
F[0]=true;
For I=1 to n do begin
F1=f;
For j=w[I] to v do
If f[j-w[I]] then f1[j]=true;
F=f1;
End;
B.求可以放入的最大价值。
C.求恰好装满的情况数。
(2)每个背包可使用任意次:
A.求最多可放入的重量。
状态转移方程为
f[I,j]=max{f[i-w[j]
B.求可以放入的最大价值。
USACO 1.2 Score Inflation
进行一次竞赛,总时间T固定,有若干种可选择的题目,每种题目可选入的
数量不限,每种题目有一个ti(解答此题所需的时间)和一个si(解答此题所得
的分数),现要选择若干题目,使解这些题的总时间在T以内的前提下,所得的
总分最大,求最大的得分。
易想到:
f[i,j] = max { f [i- kw[j], j-1] + kv[j] } (0 =k = i div w[j])
其中f[i,j]表示容量为i时取前j种背包所能达到的最大值。
优化:
Begin
FillChar(problem,SizeOf(problem),0);
Assign(Input,'inflate.in');
Reset(Input);
Readln(M,N);
For i=1 To N Do
With problem[i] Do
Readln(point,time);
Close(Input);
FillChar(f,SizeOf(f),0);
For i=1 To M Do
For j=1 To N Do
If i-problem[j].time =0 Then
Begin
t=problem[j].point+f[i-problem[j].time];
If t f[i] Then f[i]=t;
End;
Assign(Output,'inflate.out');
Rewrite(Output);
Writeln(f[M]);
Close(Output);
End.
C.求恰好装满的情况数。
Ahoi2001 Problem2
求自然数n本质不同的质数和的表达式的数目。
思路一,生成每个质数的系数的排列,在一一测试,这是通法。
procedure try(depinteger);
var i,jinteger;
begin
cal; {此过程计算当前系数的计算结果,now为结果}
if now n then exit; {剪枝}
if dep=l+1 then begin {生成所有系数}
cal;
if now=n then inc(tot);
exit;
end;
for i=0 to n div pr[dep] do begin
xs[dep]=i;
try(dep+1);
xs[dep]=0;
end;
end;
思路二,递归搜索效率较高
procedure try(dep,restinteger);
var i,j,xinteger;
begin
if (rest =0) or (dep=l+1) then begin
if rest=0 then inc(tot);
exit;
end;
for i=0 to rest div pr[dep] do
try(dep+1,rest-pr[dep]i);
end;
思路三:可使用动态规划求解
USACO1.2 money system
V个物品,背包容量为n,求放法总数。
转移方程:
Procedure update;
var j,kinteger;
begin
c=a;
for j=0 to n do
if a[j] 0 then
for k=1 to n div now do
if j+nowk =n then inc(c[j+nowk],a[j]);
a=c;
end;
{main}
begin
read(now); {读入第一个物品的重量}
i=0; {a[i]为背包容量为i时的放法总数}
while i =n do begin
a[i]=1; inc(i,now); end;
{定义第一个物品重的整数倍的重量a值为1,作为初值}
for i=2 to v do
begin
read(now);
update; {动态更新}
end;
writeln(a[n]);
End.
0/1背包问题的动态规划法相关推荐
- 0/1背包问题---C++动态规划法
[问题] 给定n种物品和一个背包,物品i(1≤i≤n)的重量是,其价值为,背包容量为,对于每种物品只有两种选择:装入背包或者不装入背包.如何选择装入背包的物品,使得装入背包中物品的总价值最大? [想法 ...
- 动态规划法改进:用序偶法求0/1背包问题
动态规划法改进:用序偶法求0/1背包问题 1.问题 2.方法 3.实现代码 序偶法求0/1背包问题(动态规划法改进版) by 孙琨SealSun at UCAS 2015.11.20 #include ...
- 算法设计与分析实验二:动态规划法实现TSP问题和0/1背包问题
[实验目的] 1.熟练掌握动态规划思想及教材中相关经典算法. 2.使用动态规划法编程,求解0/1背包问题和TSP问题. TSP问题 一.实验内容: TSP问题是指旅行家要旅行n个城市,要求每个城市经历 ...
- C语言动态规划法解决0/1背包问题(详细解答)
动态规划法解决0/1背包问题(详细解答) 首先让我们回顾一下动态规划法的使用规则: 一..动态规划法的实现思路: 1.划分子问题:将元问题分解为若干个子问题,每一个子问题对应一个决策,并且子问题之间具 ...
- 动态规划法求解0/1背包问题
问题描述 有n个重量分别为{w1,w2,-,wn}的物品,它们的价值分别为{v1,v2,-,vn},给定一个容量为W的背包. 设计从这些物品中选取一部分物品放入该背包的方案,每个物品要么选中要么不选中 ...
- 0/1背包问题——动态规划方法
1.定义 动态规划:把多阶段过程转化为一系列单阶段问题,利用各阶段之间的关系,逐个求解. 2.求解步骤 (1)找到状态转化条件 (2)归纳状态转移方程 (3)定义初始条件值 3.实例解析--0/1背包 ...
- 0/1背包问题-----回溯法求解
问题描述 有n个物品和一个容量为c的背包,从n个物品中选取装包的物品.物品i的重量为w[i],价值为p[i].一个可行的背包装载是指,装包的物品总重量不超过背包的重量.一个最佳背包装载是指,物品总价值 ...
- 0/1背包问题-----动态规划求解
问题描述 有n个物品和一个容量为c的背包,从n个物品中选取装包的物品.物品i的重量为w[i],价值为p[i].一个可行的背包装载是指,装包的物品总重量不超过背包的重量.一个最佳背包装载是指,物品总价值 ...
- 【动态规划】0/1背包问题
问题 H: [动态规划]0/1背包问题 时间限制: 1 Sec 内存限制: 64 MB 提交: 152 解决: 95 [提交] [状态] [讨论版] [命题人:admin] 题目描述 张琪曼和李旭 ...
- 分枝定界法解0/1背包问题
分枝定界法解0/1背包问题 关键词:分支定界.0-1背包 分枝定界法简介 分枝定界法按照树形结构来组织解空间,展开节点后,有两种策略: 策略一.把节点加入 FIFO 队列当中: 策略二.把节点加入到堆 ...
最新文章
- 【C#】事件,委托3点详解
- 如何从Windows远程上传文件到Linux(例如CentOS 7)
- ASP.NET Core 开源项目 nopCommerce,一款沉淀13年的电商开源佳作!
- .NET 泛型,详细介绍
- Google和微软哪个更可怕?
- Tomcat下使用Log4j 接管 catalina.out 日志文件生成方式
- 关于Objective-C 对象release操作的一个小问题探讨
- 校门外的树和memset
- JVS公众号登陆配置
- 永久删除计算机文件怎么操作步骤,如何彻底删除电脑中的文件 永久删除文件方法...
- Mac安装mysql最简单
- Unity2019学习:常用功能--Unity UI的交互游戏对象
- 三阶齐次线性方程求通解_三阶常系数线性微分方程特解的简单求法
- Double的compareTo
- 【无标题】【2023最新版】超详细Sqlmap安装保姆级教程,SQL注入使用指南,收藏这一篇就够了
- 嵌入式项目实战——基于QT的视频监控系统设计(三)
- 5分钟搞懂MySQL - 行转列
- 2019年浙江宁波宁海县下属事业单位第二批招聘教师公告(97人)
- 口袋服务器最新版,我的世界口袋版
- 关于EFI启动安装系统的一些心得
热门文章
- Spring框架 初步入门猜想
- 不使用第三个变量的情况下,实现两个变量间的互换。
- [转]常见的软件版本编号及命名
- 【转】Laplace 算子
- Codeforces 620E New Year Tree
- File(File f, String child) File(String parent, String child)
- 通俗易懂的信息熵与信息增益(IE, Information Entropy; IG, Information Gain)
- 配置Pylint for Python3.5
- android dialog转layout
- Web API 2 对于 Content-Length 要求严格