首页 > 编程语言 >背包问题算法

背包问题算法

时间:2023-11-24 18:01:20浏览次数:33  
标签:遍历 weight int 问题 算法 背包 物品 dp

01背包问题

01背包是一种动态规划问题。动态规划的核心就是状态转移方程

有一个容量为V的背包,还有n个物体。现在忽略物体实际几何形状,我们认为只要背包的剩余容量大于等于物体体积,那就可以装进背包里。每个物体都有两个属性,即体积w和价值v。 问:如何向背包装物体才能使背包中物体的总价值最大?

二维数组解法

f[i][j] = max(f[i - 1][j], f[i - 1][j - w[i]] + v[j])

i:选用前i种物品

j:背包容量为j

dp[i][j] 表示从下标为[0,i]的物品里任意取,放进容量为j的背包,价值总和最大是多少

for (int j = 0 ; j < weight[0]; j++) {  // 如果把dp数组预先初始化为0了,这一步就可以省略。
    dp[0][j] = 0;
}
// 正序遍历
for (int j = weight[0]; j <= bagWeight; j++) {
    dp[0][j] = value[0];
}

有两个遍历维度:1. 物品背包容量。 2. 所以要确定遍历顺序。

两者都可以,先遍历物品好理解。

先遍历物品,然后遍历背包重量:

// weight数组的大小 就是物品个数
for(int i = 1; i < weight.size(); i++) { // 遍历物品
    for(int j = 0; j <= bagWeight; j++) { // 遍历背包容量
        if (j < weight[i]) dp[i][j] = dp[i - 1][j]; // 如果不能放下物品i
        else dp[i][j] = max(dp[i - 1][j], dp[i - 1][j - weight[i]] + value[i]);
    }
}

先遍历背包,再遍历物品:

// weight数组的大小就是物品个数
for(int j = 0; j <= bagWeight; j++) { // 遍历背包容量
    for(int i = 1; i < weight.size(); i++) { // 遍历物品
        if (j < weight[i]) dp[i][j] = dp[i - 1][j];
        else dp[i][j] = max(dp[i - 1][j], dp[i - 1][j - weight[i]] + value[i]);
    }
}

为什么两种顺序都可以?要理解递归的本质和递推的方向。那么先遍历物品,再遍历背包的过程和先遍历背包,再遍历物品,dp[i][j]所需要的数据就是左上角,根本不影响dp[i][j]公式的推导。

lpcg112l.png

一维dp数组解决01背包问题

dp[j] = max(dp[j], dp[j - weight[i]] + value[i]);

一维dp数组如何初始化

dp[j]表示:容量为j的背包,所背的物品价值可以最大为dp[j],那么dp[0]就应该是0,因为背包容量为0所背的物品的最大价值就是0。

假设物品价值都是大于0的,所以dp数组初始化的时候,都初始为0就可以了。

一维dp数组遍历顺序

for(int i = 0; i < weight.size(); i++) { // 遍历物品
    for(int j = bagWeight; j >= weight[i]; j--) { // 遍历背包容量
        dp[j] = max(dp[j], dp[j - weight[i]] + value[i]);
    }
}

lpcg1b0w.png

  • 01背包中二维dp数组的两个for遍历的先后循序是可以颠倒,一维dp数组的两个for循环先后循序一定是先遍历物品,再遍历背包容量。

  • 和二维dp的写法中,遍历背包的顺序是不一样的!一维dp遍历的时候,背包容量是从大到小:倒序遍历是为了保证物品i只被放入一次!

为什么一维dp数组必须倒序遍历:

如图,红线为数据更新来源,蓝线数据更新顺序:当前状态源于左上角同背包容量方向

  • 如果先遍历物品再遍历背包且正序遍历,就会使一些同背包容量方向的数据被覆盖掉。
  • 如果先遍历背包再遍历物品,

完全背包问题

完全背包依然是一个动态规划问题

有N件物品和一个最多能背重量为W的背包。第i件物品的重量是weight[i],得到的价值是value[i] 。每件物品都有无限个(也就是可以放入背包多次),求解将哪些物品装入背包里物品价值总和最大。

完全背包和01背包问题唯一不同的地方就是,每种物品有无限件

01背包和完全背包唯一不同就是体现在遍历顺序上,所以针对遍历顺序进行分析。

// 先遍历物品,再遍历背包
int[] dp = new int[bagWeight + 1];
for (int i = 0; i < weight.length; i++){
    for (int j = 1; j <= bagWeight; j++){
        if (j - weight[i] >= 0){
            dp[j] = Math.max(dp[j], dp[j - weight[i]] + value[i]);
        }
    }
}

为什么遍历物品在外层循环,遍历背包容量在内层循环?

01背包中二维dp数组的两个for遍历的先后循序是可以颠倒,一维dp数组的两个for循环先后循序一定是先遍历物品,再遍历背包容量。

完全背包中,对于一维dp数组来说,其实两个for循环嵌套顺序无所谓:因为物品可以重复装。也就是说dp[j]是根据下标j之前所对应的dp[j]计算出来的。 只要保证下标j之前的dp[j]都是经过计算的就可以。完全背包中,两个for循环的先后循序,都不影响计算dp[j]所需要的值。

为什么是正序遍历而不是倒序遍历了?

因为在完全背包中,由于物品可以重复装填,那么就需要相同物品在不同背包容量时重复利用前面的数据。

lpcg1l2l.png

先遍历背包再遍历物品:

// 先遍历背包,再遍历物品
for(int j = 0; j <= bagWeight; j++) { // 遍历背包容量
    for(int i = 0; i < weight.size(); i++) { // 遍历物品
        if (j - weight[i] >= 0) dp[j] = max(dp[j], dp[j - weight[i]] + value[i]);
    }
}

lpcg1s8x.png

先遍历物品,再遍历背包:

// 先遍历物品,再遍历背包
int[] dp = new int[bagWeight + 1];
for (int i = 0; i < weight.length; i++){
    for (int j = 1; j <= bagWeight; j++){
        if (j - weight[i] >= 0){
            dp[j] = Math.max(dp[j], dp[j - weight[i]] + value[i]);
        }
    }
}

连续背包问题

连续背包问题是指有一个容量为V的背包和n个物品,每个物品有一个体积v [i]和一个价值w [i],现在要将这些物品放入背包中,使得背包中物品的总价值最大。 与0/1背包问题不同的是,每个物品可以被放入多次,即是连续的。

解题思路:单位价值优先使用贪心算法

① 按价值密度(w/v)进行非递增排序

② 按照排序先后,往里装。单个物品装完时换下一个,直到背包填满为止

标签:遍历,weight,int,问题,算法,背包,物品,dp
From: https://www.cnblogs.com/moyiv/p/17854416.html

相关文章

  • unity 精灵图集(Sprite Atlas)使用以及带来的问题
    1、图集的使用参考https://zhuanlan.zhihu.com/p/4561013732、注意点unity中设置必须是图集的设置TightPacking选项取消,若打勾切图会有问题。......
  • AI智能检测算法与LiteCVR平台铁路沿线周界入侵防护方案
    在现在铁路视频监控系统基础上,结合AI人工智能技术,通过智能算法判断铁路延线相关风险,形成“互联网+专网”常态护路联防模式,实现铁路护路“一键可控”,中心与值勤人员动态联防新格局。铁路周界的入侵防范主要包括人防、物防、技防模式。其中,人防是利用人工巡逻的方式对铁路关键位置......
  • 遇到了vue3 刷新问题
     index.d762f427.js:3[Vuewarn]:Unhandlederrorduringexecutionofschedulerflush.ThisislikelyaVueinternalsbug.Pleaseopenanissueathttps://new-issue.vuejs.org/?repo=vuejs/coreat<Tags>at<HomeonVnodeUnmounted=fn<onVnodeU......
  • 关于npm的问题整理
    npminstall提示权限不足Error:EPERM:operationnotpermitted,unlinkXXX原文[npminstall提示权限不足Error:EPERM:operationnotpermitted,unlinkXXX_npminstall--no-optional-CSDN博客]......
  • 视频监控中的智能算法与计算机视觉技术
    智能视频监控是一种基于人工智能技术的监控系统,它能够通过对图像和视频数据进行分析,自动识别目标物体、判断其行为以及进行异常检测等功能,从而实现对场景的智能化监管。以下是常见的一些用于智能视频监控的算法:1、人脸识别技术人脸识别技术是智能监控中十分常见的智能分析技术......
  • pip更新失败问题
    在使用pip安装python插件时,发现pip版本过低导致安装失败,于是使用命令pipinstall--upgradepip更新pip,结果出现如下错误,多次尝试都无法更新成功。失败原因是网络原因导致的,最终找到方法,使用本地镜像更新pip,更新成功。本地镜像更新命令如下:python-mpipinstall--upgradepip......
  • 羚通视频智能分析平台:安全帽佩戴检测识别系统与智能监控安全帽识别算法
    在现代工业生产中,安全生产是每个企业都必须重视的问题。其中,工人是否正确佩戴安全帽是一个重要的环节。为了解决这个问题,羚通视频智能分析平台推出了一款安全帽佩戴检测识别系统,通过智能监控安全帽识别算法,实现了对工人是否佩戴安全帽的自动检测和识别。一、羚通视频......
  • 羚通视频智能分析平台:人员闯入算法检测与入侵识别报警系统
    在当今社会中,安全问题已经成为人们关注的焦点。无论是家庭、企业还是公共场所,都需要有一套完善的安全防范系统来保障人们的生命财产安全。随着科技的不断发展,视频监控系统已经从传统的模拟监控升级为数字化、网络化的智能监控系统。其中,羚通视频智能分析平台的人员闯入算法检测和人......
  • 【算法】裴蜀定理
    裴蜀定理在数论中,裴蜀等式(英语:Bézout'sidentity)或裴蜀定理(Bézout'slemma)(或称贝祖等式)是一个关于最大公约数(或最大公约式)的定理。裴蜀定理得名于法国数学家艾蒂安·裴蜀,说明了对任何整数\(a\)和\(b\)和\(m\),关于未知数\(x\)和\(y\)的线性丢番图方程(称为裴蜀定理)\(a......
  • 羚通视频智能分析平台:人员闯入算法检测与入侵识别报警系统
    在当今社会中,安全问题已经成为人们关注的焦点。无论是家庭、企业还是公共场所,都需要有一套完善的安全防范系统来保障人们的生命财产安全。随着科技的不断发展,视频监控系统已经从传统的模拟监控升级为数字化、网络化的智能监控系统。其中,羚通视频智能分析平台的人员闯入算法检测和......