背包问题九讲笔记_完全背包[通俗易懂]

背包问题九讲笔记_完全背包[通俗易懂]摘自TianyiCui童鞋的《背包问题九讲》,稍作修改,方便理解。本文包含的内容:———————————————完全背包问题描述已知:有一个容量为V的背包和N件物品,第i件物品的重量是weight[i],收益是cost[i]。条件:每种物品都有无限件,能放多少就放多少。问题:在不超

大家好,又见面了,我是你们的朋友全栈君。

摘自Tianyi Cui童鞋的《背包问题九讲》,稍作修改,方便理解。

本文包含的内容:

<1> 问题描述

<2> 基本思路(直接扩展01背包的方程)

<3> 转换为01背包问题求解(直接利用01背包)

<4> O(VN)的算法

———————————————

1、问题描述

已知:有一个容量为V的背包和N件物品,第i件物品的重量是weight[i],收益是cost[i]。

条件:每种物品都有无限件,能放多少就放多少。

问题:在不超过背包容量的情况下,最多能获得多少价值或收益

举例:物品个数N = 3,背包容量为V = 5,则背包可以装下的最大价值为40.

背包问题九讲笔记_完全背包[通俗易懂]

———————————————-

2、基本思路(直接扩展01背包的方程)

由于本问题类似于01背包问题,在01背包问题中,物品要么取,要么不取,而在完全背包中,物品可以取0件、取1件、取2件…直到背包放不下位置。因此,可以直接在01背包的递推式中扩展得到。

f[i][v]:表示前i件物品放入容量为v的容量中时的最大收益
递推式:
f[i][v] = max(f[i - 1][v],f[i - K * weight[i]] + K * Value[i]); 其中  1 <= K * weight[i] <= v,(v指此时背包容量)
//初始条件
f[0][v] = 0;
f[i][0] = 0;

代码:

#include <iostream>
#include <assert.h>
using namespace std;
/*
f[i][v]:前i件物品放入背包容量为v的背包获得的最大收益

f[i][v] = max(f[i - 1][v],f[i - 1][v - k * Wi] + k * Vi,其中 1<=k<= v/Wi)

边界条件
f[0][v] = 0;
f[i][0] = 0;
*/

const int N = 3;
const int V = 5;
int weight[N + 1] = {0,3,2,2};
int Value[N + 1] = {0,5,10,20};

int f[N + 1][V + 1] = {0};

int Completeknapsack()
{
	//边界条件
	for (int i = 0;i <= N;i++)
	{
		f[i][0] = 0;
	}
	for (int v = 0;v <= V;v++)
	{
		f[0][v] = 0;
	}
	//递推
	for (int i = 1;i <= N;i++)
	{
		for (int v = 1;v <= V;v++)
		{
			f[i][v] = 0;
			int nCount = v / weight[i];
			for (int k = 0;k <= nCount;k++)
			{
				f[i][v] = max(f[i][v],f[i - 1][v - k * weight[i]] + k * Value[i]);
			}
		}
	}
	return f[N][V];
}

int main()
{
	cout<<Completeknapsack()<<endl;
	system("pause");
	return 1;
}

复杂度分析:

程序需要求解N*V个状态,每一个状态需要的时间为O(v/Weight[i]),总的复杂度为O(NV*Σ(V/c[i]))。

代码优化:

完全背包问题有一个很简单有效的优化,是这样的:若两件物品i、j满足c[i]<=c[j]且w[i]>=w[j],则将物品j去掉,不用考虑。

即,如果一个物品A是占的地少且价值高,而物品B是占地多,但是价值不怎么高,那么肯定是优先考虑A物品的。

这里代码略。

———————————————-

3、转换为01背包问题求解(直接利用01背包)

思路 1、完全背包的物品可以取无限件,根据背包的总容量V和第i件物品的总重量Weight[i],可知,背包中最多装入V/Weight[i](向下取整)件该物品。因此可以直接改变第i件物品的总个数,使之达到V/Weight[i](向下取整)件,之后直接利用01背包的思路进行操作即可。

举例:物品个数N = 3,背包容量为V = 5。

拆分之前的物品序列:

背包问题九讲笔记_完全背包[通俗易懂]

拆分之后的物品序列:

背包问题九讲笔记_完全背包[通俗易懂]

根据上述思想:在背包的最大容量(5)中,最多可以装入1件物品一,因此不用扩展物品一。最多可以装入2件物品二,因此可以扩展一件物品二。同理,可以扩展一件物品三。

时间复杂度的分析:O(NNew*V),其V表示扩展前背包容量,NNew表示扩展后物品的个数,NNew = Σ(V/Weight[i](向下取整))

思路 2、对物品进行拆分时,拆成二进制的形式。

具体思路:把第i种物品拆成费用为weight[i]*2^k、价值为w[i]*2^k的若干件物品,其中k满足weight[i]*2^k<=V。

思路:这是二进制的思想,因为不管最优策略选几件第i种物品,总可以表示成若干个2^k件物品的和。

这样把每种物品拆成O(log V/weight[i])件物品,是一个很大的改进。

举例:物品个数N = 3,背包总容量为V = 5。
拆分之前的物品序列:

背包问题九讲笔记_完全背包[通俗易懂]

拆分之后的物品序列:

背包问题九讲笔记_完全背包[通俗易懂]

为了和前面的例子保持一致,这里才用之前的例子,但是这个例子没有更好的说明二进制的拆分方法拆分的物品个数会少写。

假设物品A的重量为2,收益为3,背包的总重量为20。

根据第一种拆分,可以拆成10个物品,每一个物品的重量为2,收益为3

根据第二种拆分方法,可以拆成4个物品,分别是物品一(重量为1*2,收益为3),物品二(重量为2*2,收益为6),物品三(重量为4*2,收益为12),物品四(重量为8*2,收益为24)。

时间复杂度的分析:O(NNEW*V),其中V表示扩展前背包容量,NNew表示扩展后物品的个数,NNew = Σ(log V/weight[i](向下取整))

代码:

#include <iostream>
#include <vector>
#include <assert.h>
using namespace std;
/*
f[v]:表示第i件物品放入容量为v的背包后,获得的最大容量
f[v] = max(f[v],f[v - weight[i]] + value[i]);
初始条件:f[0] = 0;
*/

const int N = 3;
const int V = 20;//5
int weight[N + 1] = {0,3,2,2};
int Value[N + 1] = {0,5,10,20};

int NNew = 0;
vector<int> weightVector;
vector<int> Valuevector;
int f[V + 1] = {0};
/*拆分物品*/
void SplitItem()
{
	//从1开始
	weightVector.push_back(0);
	Valuevector.push_back(0);
	//开始拆分
	int nPower = 1;
	for (int i = 1;i <= N;i++)
	{
		nPower = 1;
		while (nPower * weight[i] <= V)
		{
			weightVector.push_back(nPower * weight[i]);
			Valuevector.push_back(nPower * Value[i]);
			nPower <<= 1;
		}
	}
}

int Completeknapsack()
{
	//拆分物品
	SplitItem();
	//转化为01背包处理
	NNew = weightVector.size() - 1;//多加了一个0,要减去

	for (int i = 1;i <= NNew;i++)//物品个数变化
	{
		for (int v = V;v >= weightVector[i];v--)//背包容量仍是V
		{
			f[v] = max(f[v],f[v - weightVector[i]] + Valuevector[i]);
		}
	}

	return f[NNew];
}
int main()
{
	cout<<Completeknapsack()<<endl;
	system("pause");
	return 1;
}

4、O(VN)的算法

伪代码

for (int i = 1;i <= N;i++)
{
	for (int v = weight[i];v <= V;v++)
	{
		f[v] = max(f[v],f[v - weight[i]] + Value[i]);
	}
}

分析:这和01背包的伪代码很相似,在01背包的代码中,v变化的区间是逆序循环的,即[V,Weight[i]]。而这里,v变化的区间是顺序循环的,即为[Weight[i],V]。

原因:

再次给出定义:

f[i][v]表示把前i件物品放入容量为v的背包时的最大代价。

f[i-1][v-c[i]]表示把前i – 1件物品放入容量为v的背包时的最大代价.

在01背包中,v变化的区间是逆序循环的原因要保证由状态f[i-1][v-c[i]]递推状态f[i][v]时,f[i-1][v-c[i]]没有放入第i件物品之后,在第i循环时,放入一件第i件物品。

01背包的方程:

f[i][v] = max(f[i - 1][v],f[i - 1][v - weight[i]] + Value[i])  

在完全背包中,v变化的区间是顺序循环的原因完全背包的特点是每种物品可选无限件,在求解加选第i种物品带来的收益f[i][v]时,在状态f[i][v-c[i]]中已经尽可能多的放入物品i了,此时在f[i][v-c[i]]的基础上,我们可以再次放入一件物品i,此时也是在不超过背包容量的基础下,尽可能多的放入物品i。

完全背包的方程:

f[i][v] = max(f[i - 1][v],f[i][v - weight[i]] + Value[i]);

举例:

物品个数N = 3,背包总容量为V = 5。

物品信息:

背包问题九讲笔记_完全背包[通俗易懂]

完全背包:

背包问题九讲笔记_完全背包[通俗易懂]

分析:

i = 2,表示正在处理第2件物品。在求解f[2][4]时,如果要计算把第2件物品放入背包后的代价时,我们需要知道f[2][2],此时f[2][2]中已经尽全力放入第2件物品了(已经放入一件)。此时此刻还可以在放入一件第2件物品,在背包容量为4时,最多可以放入两件第二件物品。

总结下,f[i][v]:表示在背包容量为v时,尽全力的放入第i件物品的代价。f[i][v – weight[i]]:表示在背包容量为v – weight[i]时,尽全力的放入第i件物品的代价。因此由f[i][v – weight[i]]转换为f[i][v]时,也是在f[i][v – weight[i]]的基础上有加入了一件物品i。

为了节省保存状态的空间,可以直接使用一维数组保存状态。

代码:迭代方程:f[i][v] = max(f[i – 1][v],f[i][v – weight[i]] + Value[i]);

#include <iostream>
#include <vector>
#include <assert.h>
using namespace std;
const int N = 3;
const int V = 5;//5
int weight[N + 1] = {0,3,2,2};
int Value[N + 1] = {0,5,10,20};

int f[N + 1][V + 1] = {0};

int Completeknapsack()
{
	//初始化
	for (int i = 0;i <= N;i++)
	{
		f[i][0] = 0;
	}
	for (int v = 0;v <= V;v++)
	{
		f[0][v] = 0;
	}
	for (int i = 1;i <= N;i++)
	{
		for (int v = weight[i];v <= V;v++)
		{
			f[i][v] = max(f[i - 1][v],f[i][v - weight[i]] + Value[i]);
		}
	}
	return f[N][V];
}

int main()
{
	cout<<Completeknapsack()<<endl;
	system("pause");
	return 1;
}

代码:迭代方程:
f[v] = max(f[v],f[v – weight[i]] + Value[i]);

#include <iostream>
using namespace std;
const int N = 3;
const int V = 5;//5
int weight[N + 1] = {0,3,2,2};
int Value[N + 1] = {0,5,10,20};

int f[V + 1] = {0};

int Completeknapsack()
{
	f[0] = 0;
	for (int i = 1;i <= N;i++)
	{
		for (int v = weight[i];v <= V;v++)
		{
			f[v] = max(f[v],f[v - weight[i]] + Value[i]);
		}
	}
	return f[V];
}
int main()
{
	cout<<Completeknapsack()<<endl;
	system("pause");
	return 1;
}

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/158689.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • python实现QQ和微信刷屏[通俗易懂]

    python实现QQ和微信刷屏[通俗易懂]python实现QQ和微信刷屏看过一些用来刷屏的程序,要么就只能刷屏QQ,要么就只能刷屏微信,今天博主就来把它一起实现了,而且用法超简单的哦!!!,希望可以帮助到你!废话不多说,先上代码,然后再进行详细介绍!!!frompynputimportmouse,keyboardfromtkinterimport*importtkinter.filedialogimporttimeroot=Tk()root.title(“信息刷屏”)root.geometry(“550×200

    2022年6月11日
    91
  • ingress什么意思_ingress负载均衡

    ingress什么意思_ingress负载均衡k8sIngress介绍Http代理Https代理Ingress介绍我们已经知道,Service对集群之外暴露服务的主要方式有两种:NodePort和LoadBalancer,但是这两种方式,都有一定的缺点:NodePort方式的缺点是会占用很多集群机器的端口,那么当集群服务变多的时候,这个缺点就愈发明显。LoadBalancer的缺点是每个Service都需要一个LB,浪费,麻烦,并且需要kubernetes之外的设备的支持。基于这种现状,kubernetes提供了Ingress资源对象,I

    2022年8月11日
    5
  • MySQL允许root远程登录[通俗易懂]

    MySQL允许root远程登录[通俗易懂]新安装的数据库只能localhost访问??是不是很苦逼下面介绍如何允许远程访问root1.“试一下”能否远程登录&gt;mysql-uroot-p-h10.0.42.180答案是否定的。那就开始进行设置吧2.登录数据库,默认本地访问&gt;mysql-uroot-p3.切换mysql数据库mysql&gt;usem…

    2022年6月17日
    30
  • 极大似然、最小二乘和梯度下降

    极大似然、最小二乘和梯度下降

    2021年11月19日
    38
  • office2013产品密钥_office365激活密钥

    office2013产品密钥_office365激活密钥HV7BJ-R6GCT-VHBXD-YH4FD-GTH2T87XPX-M3D6G-W4D39-VKVKR-DB8C7HM7R6-FP6QB-XTDC3-MT442-FVPKMXJBYM-62WK4-RCT9Y-XG3HQ-M2CMKHMYY4-TR62Q-9TT76-BDBHK-WPRPTHV7BJ-R6GCT-VHBXD-YH4FD-GTH2Thttp://zhida…

    2022年10月9日
    0
  • 如何将深度学习的float32图像转为Unit8格式以方便cv2使用

    如何将深度学习的float32图像转为Unit8格式以方便cv2使用在使用Pyside2中的QImage处理深度学习模型生成的图片时,需要将float32的图像转为Unit8格式,再使用cv2处理。一开始使用网上的其他教程,如下: #模型生成 G_recon=G(self.content,True) #将(1,3,256,256)尺寸的转为(256,256,3)G_recon=((G_recon[0].cpu().detach().numpy().transpose(1,2,0)+1)/2)

    2022年9月15日
    0

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号