【蓝桥杯训练】背包问题
创始人
2025-05-30 23:18:33
0

1. 01背包问题

image-20221008212851875

1.1 版本1 二维

(1)状态f[i][j]定义:前 ii 个物品,背包容量 jj 下的最优解(最大价值):

当前的状态依赖于之前的状态,可以理解为从初始状态f[0][0] = 0开始决策,有 NN 件物品,则需要 NN 次决 策,每一次对第 ii 件物品的决策,状态f[i][j]不断由之前的状态更新而来。
(2)当前背包容量不够(j < v[i]),没得选,因此前 ii 个物品最优解即为前 i−1i−1 个物品最优解:

对应代码:f[i][j] = f[i - 1][j]。
(3)当前背包容量够,可以选,因此需要决策选与不选第 ii 个物品:

选:f[i][j] = f[i - 1][j - v[i]] + w[i]。
不选:f[i][j] = f[i - 1][j] 。
我们的决策是如何取到最大价值,因此以上两种情况取 max() 。

dp数组:所有包括第i个物品且体积不超过j的物品的集合。

核心思想,dp数组的分类处理,是否包含第i件物品,如果不包含

dp[i][j]=dp[i-1][j];

需要保证背包可以装下第i个物品

dp[i][j]=Math.max(dp[i][j],dp[i-1][j-v[i]]+w[i]);
import java.util.*;
public class Main{public static void main(String[] args){Scanner scan = new Scanner(System.in);int N = 1010;int[] v = new int[N];int[] w = new int[N];int[][] f = new int[N][N];int n = scan.nextInt();int m = scan.nextInt();for(int i = 1 ; i <= n ; i ++ ){v[i] = scan.nextInt();w[i] = scan.nextInt();}for(int i = 1 ; i <= n ; i ++ ){for(int j = 0 ; j <= m ; j ++ ){f[i][j] = f[i - 1][j]; // 左边不包含i的方案if(j >= v[i])  f[i][j] = Math.max(f[i][j] , f[i - 1][j - v[i]] + w[i]);//右边包含i的方案,f[i-1][j - v[i]] + w[i]}}System.out.println(f[n][m]);}
}

1.2 版本2 一维

将状态f[i][j]优化到一维f[j],实际上只需要做一个等价变形。

为什么可以这样变形呢?我们定义的状态f[i][j]可以求得任意合法的i与j最优解,但题目只需要求得最终状态f[n][m],因此我们只需要一维的空间来更新状态。

(1)状态f[j]定义:NN 件物品,背包容量j下的最优解。

(2)注意枚举背包容量j必须从m开始。

(3)为什么一维情况下枚举背包容量需要逆序?在二维情况下,状态f[i][j]是由上一轮i - 1的状态得来的,f[i][j]与f[i - 1][j]是独立的。而优化到一维后,如果我们还是正序,则有f[较小体积]更新到f[较大体积],则有可能本应该用第i-1轮的状态却用的是第i轮的状态。

(4)例如,一维状态第i轮对体积为 33 的物品进行决策,则f[7]由f[4]更新而来,这里的f[4]正确应该是f[i - 1][4],但从小到大枚举j这里的f[4]在第i轮计算却变成了f[i][4]。当逆序枚举背包容量j时,我们求f[7]同样由f[4]更新,但由于是逆序,这里的f[4]还没有在第i轮计算,所以此时实际计算的f[4]仍然是f[i - 1][4]。

(5)简单来说,一维情况正序更新状态f[j]需要用到前面计算的状态已经被「污染」,逆序则不会有这样的问题。

状态转移方程为:f[j] = max(f[j], f[j - v[i]] + w[i] 。

import java.util.*;
public class Main{public static void main(String[] args){Scanner scan = new Scanner(System.in);int N = 1010;int[] v = new int[N];int[] w= new int[N];int[] f = new int[N];int n = scan.nextInt();int m = scan.nextInt();for(int i = 1;i<=n;i++){v[i] = scan.nextInt();w[i] = scan.nextInt();}for(int i = 1; i<=n;i++){for(int j = m;j>=0;j--){if(j >=v[i]) f[j] = Math.max(f[j],f[j-v[i]]+w[i]);}}System.out.println(f[m]);}
}

2. 完全背包问题

image-20221008221251420

思路:

同01背包问题。区别在于01背包对于每种物品只有选或不选,这也即「01」的由来。多重背包则对于每种物品可以多次选择。

2.1 二维朴素写法

import java.util.*;
public class Main{public static void main(String[] args){int N = 1010;int[] v= new int[N];int[] w =new int[N];int[][] f = new int[N][N];Scanner scan = new Scanner(System.in);int n = scan.nextInt();int m = scan.nextInt();for(int i = 1; i <= n ;i++){v[i] = scan.nextInt();w[i] = scan.nextInt();}for(int i = 1; i <=n;i++){for(int j = 0;j<=m;j++){f[i][j] = f[i-1][j];if(j >= v[i]) f[i][j] = Math.max(f[i][j],f[i][j-v[i]]+w[i]);}}System.out.println(f[n][m]);}
}

2.2 一维优化版

这里对比01背包问题,注意下标,我们可以发现

if(j >= v[i]) dp[i][j]=Math.max(dp[i][j],dp[i-1][j-v[i]]+w[i]);//01背包
if(j >= v[i]) f[i][j] = Math.max(f[i][j],f[i][j-v[i]]+w[i]);//完全背包

仅仅是f[i]dp[i-1]的区别,所以对应一维优化

import java.util.*;
public class Main{public static void main(String[] args){Scanner scan = new Scanner(System.in);int N = 1010;int[] v = new int[N];int[] w = new int[N];int[] f = new int[N];int n = scan.nextInt();int m = scan.nextInt();for(int i = 1 ; i <= n ; i ++ ){v[i] = scan.nextInt();w[i] = scan.nextInt();}for(int i = 1 ; i <= n ; i ++ ){for(int j = m ; j >= v[i] ; j -- ){f[j] = Math.max(f[j] , f[j - v[i]] + w[i]);}}}System.out.println(f[m]);}
}

3. 多重背包问题

image-20221108110733785

分析

当 si=1 时,相当于01背包中的一件物品
当 si>1 时,相当于01背包中的多个一件物品
故我们可以死拆(把多重背包拆成01背包)

直接枚举第i件物品的选择数,满足总体积不超过j即可

朴素解法

import java.util.*;
public class Main{public static void main(String[] args){Scanner scan = new Scanner(System.in);int N = 110;int[] v = new int[N],w = new int[N],s = new int[N];int[][] f = new int[N][N];int n = scan.nextInt();int m = scan.nextInt();for(int i = 1 ; i <= n ; i ++ ){v[i] = scan.nextInt();w[i] = scan.nextInt();s[i] = scan.nextInt();}for(int i = 1 ; i <= n ; i ++ )for(int j = 0 ; j <= m ; j ++ )for(int k = 0 ; k <= s[i] && k * v[i] <= j; k ++ )f[i][j] = Math.max(f[i][j], f[i - 1][j - k * v[i]] + k * w[i]);System.out.println(f[n][m]);}
}

4. 分组背包问题

image-20221108110414313

分析:

image-20221108110457239

朴素解法

import java.util.*;
public class Main{public static void main(String[] ags){Scanner scan = new Scanner(System.in);int N = 110;int[][] v = new int[N][N];int[][] w = new int[N][N];int[] s = new int[N];int[] f = new int[N];int n = scan.nextInt();int m = scan.nextInt();for(int i = 1 ; i <= n ; i ++ ){s[i] = scan.nextInt();for(int j = 1 ; j <= s[i] ; j ++ ){v[i][j] = scan.nextInt();w[i][j] = scan.nextInt();}}for(int i = 1 ; i <= n ; i ++ ){for(int j = m ; j >= 0 ; j -- ){for(int k = 0; k <= s[i] ; k ++ ){if(j >= v[i][k])  f[j] = Math.max(f[j], f[j - v[i][k]] + w[i][k]);}}}System.out.println(f[m]);}
}
           for(int k = 0; k <= s[i] ; k ++ ){if(j >= v[i][k])  f[j] = Math.max(f[j], f[j - v[i][k]] + w[i][k]);}}}System.out.println(f[m]);
}

}


相关内容

热门资讯

期货市场沉淀资金创7783亿历... 近期期货市场资金流入呈现加速态势,沉淀资金总量达到近7783亿元的历史新高。这一数据自6月25日以来...
价格法修正草案公布,强调“反内... 政策面再出“反内卷”利好,7月24日,《中华人民共和国价格法修正草案》公开征求意见:明确不正当价格行...
原创 特... 关税风暴:特朗普贸易战的全球冲击波 8月1日,一场由美国总统特朗普发起的全球贸易战即将进入高潮。看似...
2024消费品上市公司研究报告... 《2024消费品上市公司研究报告》 (内容出品方:和君咨询x新华网) 报告共计:40页 本研报通过对...
特斯拉Q2营收下滑12%,马斯... 特斯拉于7月24日公布了2025年第二季度财报,数据显示总营收降至225.0亿美元,同比下滑12%;...
衣食住行跟我逛|“今年更甜!”... 盛夏时节,誉称为“紫水晶”的黑葡萄迎来最佳赏味期。近日,扬子晚报/紫牛新闻记者走访苏州市场发现,多数...
棕榈油与油菜籽:印尼库存降,马... 【印尼5月末棕榈油库存下滑,马棕7月上旬增产,加菜籽预估调整】印尼棕榈油协会数据显示,因出口激增,印...
康佳易主华润 半导体业务整合成... 康佳集团股份有限公司是深圳首家营业收入超百亿元的工业企业。 来源:康佳集团 靴子落地。在长达三个多月...
英特尔营收超预期,宣布裁员,C... 英特尔数据中心+AI收入超预期难掩盈利困境,英特尔CEO未能证明公司将扭亏为盈,股价应声下跌。 周四...
我国新药好药呈现快速增长态势:... 央视网消息:国家药监局最新统计显示,我国上半年批准创新药43个,同比增长59%,接近2024年批准创...
赛峰集团宣布完成对柯林斯宇航飞... 赛峰集团宣布完成对柯林斯宇航(Collins Aerospace)飞行控制与作动业务的收购。该业务为...
中国最大农业互联网公司即将登录... 来源:ACN亚太商讯 近日,中国领先的农产品B2B数字化服务公司一亩田集团向美国证券交易委员会(SE...
常熟银行股价微跌0.54% 拟... 截至2025年7月24日收盘,常熟银行股价报7.37元,较前一交易日下跌0.54%,成交额3.16亿...
股市必读:湖南黄金(00215... 截至2025年7月24日收盘,湖南黄金(002155)报收于18.58元,上涨0.27%,换手率2....
股票行情快报:浙江众成(002... 证券之星消息,截至2025年7月24日收盘,浙江众成(002522)报收于5.07元,上涨1.0%,...
“反内卷”与利率上涨 冉学东 反内卷政策持续推进,金融市场反应明显,首先就是大宗商品的牛市。 大宗商品期货市场近期出现强劲...
24日不锈钢下跌0.08%,最... 来源:新浪期货 新浪期货 根据交易所数据,截至7月24日收盘主力合约不锈钢2509,涨跌-0.08%...
稀有金属ETF(159608)... 中证网讯 7月24日,稀有金属概念股午后拉升,锂矿、稀土永磁板块走强,相关ETF领涨。截至收盘,稀有...
北交所上市公司生物谷大宗交易折... 每经讯,2025年7月24日,北交所上市公司生物谷(833266,收盘价:11.7元)发生一笔大宗交...
瑞信证券更名,北京证券重出江湖... (图片来源:视觉中国) 蓝鲸新闻7月24日讯(记者 王婉莹)从外资券商转身为国资控股券商后,瑞信证券...