回顾复习之背包DP
\(\small{(本文统一将c[i]视作cost,w[i]视作worth,下面的代码用这两个变量表示费用和价值)}\)
\(\Large\textbf{1. 01背包}\)
- \(\large\textbf{描述:}\)
有n个物品,每个物品只有一件,第i个物品体积为vi,价格为ci。现在有一个体积为V的背包,请你从n件物品里选出若干件放进背包里,使得背包里的物品价值最大。 - \(\large\textbf{思路:}\)
01背包的特点是:每种物品只有一件,可以选择放或不放。
我们可以根据此特点进行动态规划(DP),设f[i][j]表示前i件物品放入一个容量为j的背包中可以获得的最大价值,则易得状态转移方程为
dp[i][j]=max(dp[i-1][j],dp[i-1][j-c[i]]+w[i])
(详解:在“将前i个物品放入容量为j的背包中”的这个子问题中,由题意,我们只有\(\textbf{放}\)或\(\textbf{不放}\)两种选择,那么就转化为一个只与前i-1个物品有关的问题。如果不放第i件物品,那么就是“前i-1件物品放入容量为j的背包中”,最大价值为f[i-1][j];如果放第i件物品,那么就是“前i-1件物品放入剩下的容量为j-c[i]的背包中”,最大价值为f[i-1][j-c[i]]+w[i])
由此可得01背包的\(\textbf{最原始}\)代码
$\large\textbf{code}$
#include<bits/stdc++.h>
using namespace std;
#define N 1010
#define Elaina 0
int n,m,V,c[N],w[N],dp[N][N],ans=0;
void bag(){
for(int i=1;i<=n;i++){
for(int j=1;j<=V;j++){
dp[i][j]=dp[i-1][j];
if(j>=c[i]){
dp[i][j]=max(dp[i-1][j],dp[i-1][j-c[i]]+w[i]);
}
}
}
}
int main(){
cin>>n>>V;
for(int i=1;i<=n;i++){
cin>>c[i]>>w[i];
}
bag();
cout<<dp[n][V];
return Elaina;
}
以上代码的时间和空间的复杂度均为O(V*N),其中时间已经不能进一步优化了,但是空间可以
\(\textbf{滚动数组优化code}\)
code
#include<bits/stdc++.h>
using namespace std;
#define N 1010
#define Elaina 0
int n,m,V,c[N],w[N],ans=0;
int dp[2][N];//滚动数组优化 只开2行数组
void bag(){
for(int i=1;i<=n;i++){
for(int j=1;j<=V;j++){
dp[i&1][j]=dp[(i-1)&1][j];
if(j>=c[i]){
dp[i&1][j]=max(dp[i&1][j],dp[(i-1)&1][j-c[i]]+w[i]);
}
}
}
}
int main(){
cin>>n>>V;
for(int i=1;i<=n;i++){
cin>>c[i]>>w[i];
}
bag();
cout<<dp[n&1][V];
return Elaina;
}
\(\textbf{一维优化code}\)
code
#include<bits/stdc++.h>
using namespace std;
#define N 1010
#define Elaina 0
int n,m,V,c[N],w[N],ans;
int dp[N];//一维数组优化
void bag(){
for(int i=1;i<=n;i++){
for(int j=V;j>=c[i];j--){
dp[j]=max(dp[j],dp[j-c[i]]+w[i]);
}
}
}
int main(){
cin>>n>>V;
for(int i=1;i<=n;i++){
cin>>c[i]>>w[i];
}
bag();
cout<<dp[V];
return Elaina;
}
\(\Large\textbf{2. 完全背包}\)
- \(\large\textbf{描述:}\)
设有n种物品,每种物品有一个重量及一个价值。但每种物品的数量是无限的,同时有一个背包,最大载重量为M,今从n种物品中选取若干件(同一种物品可以多次选取),使其重量的和小于等于M,而价值的和为最大。 - \(\large\textbf{思路:}\)
完全背包的特点是:每种物品有无数件,可以选择放若干件或不放。
设k为取的物品的数量,依据01背包思路,易得状态转移方程为
dp[i][j]=max(dp[i-1][j],dp[i-1][j-k*c[i]]+k*w[i])
完整代码为
$\large\textbf{code}$
#include<bits/stdc++.h>
using namespace std;
#define N 10100
#define Elaina 0
int n,m,V,c[N],w[N],dp[N][N],ans=0;
void bag(){
for(int i=1;i<=n;i++){
for(int j=1;j<=V;j++){
for(int k=0;k*c[i]<=j;k++){
dp[i][j]=max(dp[i-1][j],dp[i-1][j-k*c[i]]+k*w[i]);
}
}
}
}
int main(){
cin>>V>>n;
for(int i=1;i<=n;i++){
cin>>c[i]>>w[i];
}
bag();
cout<<dp[n][V];
return Elaina;
}
\(\textbf{一维优化code}\)
code
#include<bits/stdc++.h>
using namespace std;
#define N 10100
#define Elaina 0
int n,m,V,c[N],w[N],dp[N],ans=0;
void bag(){
for(int i=1;i<=n;i++){
for(int j=c[i];j<=V;j++){
dp[j]=max(dp[j],dp[j-c[i]]+w[i]);
}
}
}
int main(){
cin>>V>>n;
for(int i=1;i<=n;i++){
cin>>c[i]>>w[i];
}
bag();
cout<<dp[V];
return Elaina;
}
\(\Large\textbf{3. 多重背包}\)
- \(\large\textbf{描述:}\)
有N种物品和一个容量为V的背包。第i ii种物品最多有p[i]件可用,每件费用是c[i],价值是w[i]。求解将哪些物品装入背包可使这些物品的费用总和不超过背包容量,且价值总和最大。 - \(\large\textbf{思路:}\)
这题目和完全背包问题很类似。基本的方程只需将完全背包问题的方程略微一改即可,因为对于第i种物品有p[i]+1种策略:取0件,取1件……取p[i]件。令dp[i][j]表示前i种物品恰放入一个容量为j的背包的最大价值,则有状态转移方程:
dp[i][j]=max(dp[i][j],dp[i-1][j-k*c[i]]+k*w[i])
$\large\textbf{code}$
#include<bits/stdc++.h>
using namespace std;
#define N 10100
#define Elaina 0
int n,m,V,c[N],w[N],dp[N][N],s[N];
void bag(){
for(int i=1;i<=n;i++){
for(int j=1;j<=V;j++){
for(int k=0;k<=s[i]&&k*c[i]<=j;k++){
dp[i][j]=max(dp[i][j],dp[i-1][j-k*c[i]]+k*w[i]);
}
}
}
}
int main(){
cin>>n>>V;
for(int i=1;i<=n;i++){
cin>>c[i]>>w[i]>>s[i];
}
bag();
cout<<dp[n][V];
return Elaina;
}
\(\textbf{一维优化code}\)
code
#include<bits/stdc++.h>
using namespace std;
#define N 1010
#define Elaina 0
int n,m,V,c[N],w[N],dp[N],s[N];
void bag(){
for(int i=1;i<=n;i++){
for(int j=V;j>=0;j--){
for(int k=0;k<=s[i]&&k*c[i]<=j;k++){
dp[j]=max(dp[j],dp[j-k*c[i]]+k*w[i]);
}
}
}
}
int main(){
cin>>n>>V;
for(int i=1;i<=n;i++){
cin>>c[i]>>w[i]>>s[i];
}
bag();
cout<<dp[V];
return Elaina;
}
\(\Large\textbf{4. 混合背包}\)
- \(\large\textbf{描述:}\)
一个旅行者有一个最多能用V公斤的背包,现在有n件物品,它们的重量分别是W1,W2,...,Wn,它们的价值分别为C1,C2,...,Cn。有的物品只可以取一次(01背包),有的物品可以取无限次(完全背包),有的物品可以取的次数有一个上限(多重背包)。求解将哪些物品装入背包可使这些物品的费用总和不超过背包容量,且价值总和最大。
(就是把前01背包、完全背包、多重背包搓起来 搓吧搓吧就出来了(^_−)☆)
直接请出代码君
code
#include<bits/stdc++.h>
using namespace std;
#define N 10100
#define inf 0x3f3f3f3f
#define Elaina 0
int idx=0,n,V,c[N],w[N],dp[N],s[N],ans=inf;
void bag(){
for (int i=1; i<=n; i++){
if(s[i]==1){
for(int j=V;j>=c[i];j--){
dp[j]=max(dp[j],dp[j-c[i]]+w[i]);
}
}
else if(s[i]==0){
for(int j=c[i];j<=V;j++){
dp[j]=max(dp[j],dp[j-c[i]]+w[i]);
}
}else{
for(int j=V;j>=0;j--){
for(int k=0;k<=s[i]&&k*c[i]<=j;k++){
dp[j]=max(dp[j],dp[j-k*c[i]]+k*w[i]);
}
}
}
}
}
int main(){
cin>>V>>n;
for(int i=1;i<=n;i++){
cin>>c[i]>>w[i]>>s[i];
}
bag();
cout<<dp[V];
return Elaina;
}
\(\Large\textbf{5. 分组背包}\)
- \(\large\textbf{描述:}\)
一个旅行者有一个最多能用V公斤的背包,现在有n件物品,它们的重量分别是W1,W2,...,Wn,它们的价值分别为C1,C2,...,Cn。这些物品被划分为若干组,每组中的物品互相冲突,最多选一件。求解将哪些物品装入背包可使这些物品的费用总和不超过背包容量,且价值总和最大 - \(\large\textbf{死路:}\)
这个问题变成了每组物品有若干种策略:是选择本组的某一件,还是一件都不选。也就是说设dp[k][j]表示前k组物品花费费用j能取得的最大价值,则有:
dp[k][j]=max(dp[k][j],dp[k-1][j-c[i]]+w[i]
\(\textbf{一维优化code}\)
code
#include<bits/stdc++.h>
using namespace std;
#define N 10100
#define ll long long
#define inf 0x3f3f3f3f
#define Elaina 0
int n,t,V,T,c[N],w[N],dp[N],g[N][N];
void bag(){
for (int i=1; i<=T; i++){
for(int j=V;j>=0;j--){
for(int k=1;k<=g[i][0];k++){
int x=g[i][k];
if(j>=c[x]){
dp[j]=max(dp[j],dp[j-c[x]]+w[x]);
}
}
}
}
}
int main(){
cin>>V>>n>>T;
for(int i=1;i<=n;i++){
cin>>c[i]>>w[i]>>t;
g[t][++g[t][0]]=i;
}
bag();
cout<<dp[V];
return Elaina;
}
回顾复习之背包DP的更多相关文章
- 复习1背包dp
背包问题是对于一个有限制的容器,一般计算可以装的物品的价值最值或数量.通常每个物品都有两个属性空间和价值,有时还有数量或别的限制条件,这个因体而异. 背包大概分成3部分,下面会细述这最经典的3种题型 ...
- 算法复习——背包dp
1.01背包 二维递推式子: 代码: ;i<=n;i++) ;x--) ][x-w[i]]+c[i],f[i-][x]); ][x]; printf("%d",f[n][m] ...
- hdu 2844 混合背包【背包dp】
http://acm.hdu.edu.cn/showproblem.php?pid=2844 题意:有n种纸币面额(a1,a2,...an),每种面额对应有(c1,c2,...cn)张.问这些钱能拼成 ...
- 背包dp整理
01背包 动态规划是一种高效的算法.在数学和计算机科学中,是一种将复杂问题的分成多个简单的小问题思想 ---- 分而治之.因此我们使用动态规划的时候,原问题必须是重叠的子问题.运用动态规划设计的算法比 ...
- hdu 5534 Partial Tree 背包DP
Partial Tree Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://acm.hdu.edu.cn/showproblem.php?pid= ...
- HDU 5501 The Highest Mark 背包dp
The Highest Mark Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://acm.hdu.edu.cn/showproblem.php?p ...
- Codeforces Codeforces Round #319 (Div. 2) B. Modulo Sum 背包dp
B. Modulo Sum Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/contest/577/problem/ ...
- noj [1479] How many (01背包||DP||DFS)
http://ac.nbutoj.com/Problem/view.xhtml?id=1479 [1479] How many 时间限制: 1000 ms 内存限制: 65535 K 问题描述 The ...
- HDU 1011 树形背包(DP) Starship Troopers
题目链接: HDU 1011 树形背包(DP) Starship Troopers 题意: 地图中有一些房间, 每个房间有一定的bugs和得到brains的可能性值, 一个人带领m支军队从入口(房 ...
- BZOJ 1004: [HNOI2008]Cards( 置换群 + burnside引理 + 背包dp + 乘法逆元 )
题意保证了是一个置换群. 根据burnside引理, 答案为Σc(f) / (M+1). c(f)表示置换f的不动点数, 而题目限制了颜色的数量, 所以还得满足题目, 用背包dp来计算.dp(x,i, ...
随机推荐
- 上下文中找不到org.springframework.boot.web.servlet.server.ServletWebServerFactory bean
1.问题 报错如下: Description: Web application could not be started as there was no org.springframework.boo ...
- html - 多次点击选中页面文字出现蓝色背景的解决方法
body{ -moz-user-select: none; /*火狐*/ -webkit-user-select: none; /*webkit浏览器*/ -ms-user-select: none; ...
- [转帖]Oracle数据库下PreparedStatementCache内存问题解决方案
https://github.com/alibaba/druid/wiki/Oracle%E6%95%B0%E6%8D%AE%E5%BA%93%E4%B8%8BPreparedStatementCac ...
- [转帖]一文读懂 HugePages(大内存页)的原理
https://juejin.cn/post/6956541214426398757 在介绍 HugePages 之前,我们先来回顾一下 Linux 下 虚拟内存 与 物理内存 之间的关系. 物理内存 ...
- [转帖]11GR2数据库审计日志自动清理
审计日志如果把SYSTEM表空间撑爆,也会导致数据库停摆,且11g默认审计是开启状态. 今天就遇到了这样的情况,写了下面脚本来实现自动清理工作,记录操作过程. TRUNCATE TABLE SYS.A ...
- [转帖]Shell中常用的date时间命令
常用FORMAT %Y YYYY格式的年份(Year) %m mm格式的月份(),01-12 %d dd格式的日期(day of month),01-31 %H HH格式的小时数(),00 ...
- [转帖]华为毕昇 JDK 8u292、11.0.11 发布!
https://baijiahao.baidu.com/s?id=1705499834793298544&wfr=spider&for=pc 2021 年 6 月 30 日,毕昇 JD ...
- [译]深入了解现代web浏览器(三)
本文是根据Mariko Kosaka在谷歌开发者网站上的系列文章https://developer.chrome.com/blog/inside-browser-part3/ 翻译而来,共有四篇,该篇 ...
- echarts饼状图自定义legend的样式付费
先看效果图 代码 <!DOCTYPE html> <html> <head> <meta charset="utf-8"> < ...
- 小记录 单选框的注意点 html中字符串拼接 el-upload手动上传 表格跳转 v-for动态添加背景色 控制label标签于文本框之间的间距
在element-ui中 单选框的v-model的值最好是一个字符串 否者可能不能够进行数据回填哈 单选框 的类型必须是字符串类型哈 在elemnet-ui中 如果你想从A页面拿到B页面中的值 可以有 ...