dp小结|背包问题
1.先放上0-1背包模板
二维数组
for(int i=1;i<=n;i++)//枚举 物品
for(int j=1;j<=V;j++)//枚举体积
//这个位置是可以正序枚举的. qwq
//一维01背包必须倒叙 emmm
//这个没错a emmm
if(j>=c[i])
f[i][j]=max(f[i-1][j],f[i-1][j-c[i]]+w[i]);//状态转移方程.
else f[i][j]=f[i-1][j].
滚动二维数组
int dp[2][10010];
int row = 0; //滚动
for (int i = 1; i < n; ++i) {
row = 1 - row;
for (int j = 1; j <= i * (i + 1) / 2; ++j) {//这里可以减少枚举
if(j>=c[i])
f[row][j]=max(f[1-row][j],f[1-row][j-c[i]]+w[i]);//状态转移方程.
else f[row][j]=f[1-row][j].
}
}
一维数组(逆序)
for(int i=1;i<=n;i++)//枚举 物品
for(int j=V;j>=c[i];j--)//枚举体积
f[j]=max(f[j],f[j-c[i]]+w[i]);//状态转移方程.
模板题:洛谷:P1048采药
“采药”这道题套模板就行了,从二维到一维优化。
2.顺便再复习记忆化搜索洛谷:P1048采药记忆化搜索做法题解
先想到dfs搜索暴力解题,就是多参数递归,出口记录搜索到的值
再想到记忆化搜索,什么时记忆化搜索?记录每一次dfs答案,免去重复计算,从而起到优化时间复杂度的作用
记忆化搜索记录什么值呢?记录每次搜索结束找到的价值(不一定最大)


如何想到可以用记忆化搜索?

3.蓝桥杯考过的记忆化搜索——2013年C++B组第9题地宫取宝
使用四维数组缓存记录。
4.蓝桥杯考过的0-1背包:2014年C++A组第十题-波动数列
这道题比较难想到0-1背包,选数字对应成拿物品填充背包,dp[i][j]表示用前i个数凑出数字j的方案数,类似题目:我记得有一道选几个数字,凑出数字n。初始化dp[0][0] = 1,第0列都等于1,dp[i~n][0] = 1
5.完全背包
与0-1背包略有不同的是,每种物品有无限多个,可重复选取。
二维数组,三层循环模板
for(int i=1;i<=n;i++)//枚举物品
for(int j=1;j<=V;j++)
for(int k=1;k<=V/c[i];k++)//我们的物品最多只能放多少件.
{
if(k*c[i]<=j)
f[i][j]=max(f[i-1][j],f[i-1][j-k*c[i]]+k*w[i]);
else
f[i][j]=f[i-1][j];
//判断条件与01背包相同.
}
一维数组,两层循环模板(顺序枚举,和0-1背包一维恰好相反)
for(int i=1;i<=n;i++)//枚举物品
for(int j=c[i];j<=V;j++)//枚举体积.注意这里是顺序/
f[j]=max(f[j],f[j-c[i]]+w[i]);//状态转移.
完全背包模板题:P1616 疯狂的采药
6.蓝桥杯考过的完全背包:2017年蓝桥杯-包子凑数
1~100000(比较大的值)
初始化dp[0] = 1
完全背包思想,推出布尔类型的dp数组,最后遍历最后dp数组最后一层为true的个数
整数划分(每个数可以用无限次)——完全背包
#include<bits/stdc++.h>
using namespace std;
const int maxn = 100005;
int n;
int dp[maxn];
int main(){
cin>>n;
dp[0] = 1;
for(int i=1;i<=n;i++){//可以用到n
for(int j=i;j<=n;j++){//完全背包 每个数字可以用无限次
dp[j] = dp[j] + dp[j-i];
}
}
cout<<dp[n]<<endl;
return 0;
}
整数划分(每个数只能用一次)——0-1背包
#include<bits/stdc++.h>
using namespace std;
const int maxn = 100005;
int n;
int dp[maxn];
int main(){
cin>>n;
dp[0] = 1;
for(int i=1;i<=n-1;i++){//不能用n
for(int j=n;j>=i;j--){//0-1背包 倒序
dp[j] = dp[j] + dp[j-i];
}
}
cout<<dp[n]<<endl;
return 0;
}
//输入6
//输出3 说明:1、5;2、4;1、2、3
7.多重背包,没做过题不好讲复习。。
多重背包问题限定了一种物品的个数,一个简单的方法,转成0-1背包。先这样写吧
#include <iostream>
using namespace std;
#define V 1000
int weight[50 + 1];
int value[50 + 1];
int num[20 + 1];
int f[V + 1];
int max(int a, int b) {
return a > b ? a : b;
}
int main() {
int n, m;
cout << "请输入物品个数:";
cin >> n;
cout << "请分别输入" << n << "个物品的重量、价值和数量:" << endl;
for (int i = 1; i <= n; i++) {
cin >> weight[i] >> value[i] >> num[i];
}
int k = n + 1;
for (int i = 1; i <= n; i++) {
while (num[i] != 1) {
weight[k] = weight[i];
value[k] = value[i];
k++;
num[i]--;
}
}
cout << "请输入背包容量:";
cin >> m;
for (int i = 1; i <= k; i++) {
for (int j = m; j >= 1; j--) {
if (weight[i] <= j) f[j] = max(f[j], f[j - weight[i]] + value[i]);
}
}
cout << "背包能放的最大价值为:" << f[m] << endl;
}
学长的dp:https://blog.csdn.net/o_ohello/article/details/89378697
dp小结|背包问题的更多相关文章
- 【dp】 背包问题
问题一:01背包 题目: [题目描述] 一个旅行者有一个最多能装 M 公斤的背包,现在有 n件物品,它们的重量分别是W1,W2,...,Wn它们的价值分别为C1,C2,...,Cn求旅行者能获得最大总 ...
- 动态规划(DP),0-1背包问题
题目链接:http://poj.org/problem?id=3624 1.p[i][j]表示,背包容量为j,从i,i+1,i+2,...,n的最优解. 2.递推公式 p[i][j]=max(p[i+ ...
- Codeforces Round #105 (Div. 2) E. Porcelain —— DP(背包问题)
题目链接:http://codeforces.com/problemset/problem/148/E E. Porcelain time limit per test 1 second memory ...
- 「10.19」最长不下降子序列(DP)·完全背包问题(spfa优化DP)·最近公共祖先(线段树+DFS序)
我又被虐了... A. 最长不下降子序列 考场打的错解,成功调了两个半小时还是没A, 事实上和正解的思路很近了,只是没有想到直接将前$D$个及后$D$个直接提出来 确实当时思路有些紊乱,打的时候只是将 ...
- [动态规划]状态压缩DP小结
1.小技巧 枚举集合S的子集:for(int i = S; i > 0; i=(i-1)&S) 枚举包含S的集合:for(int i = S; i < (1<<n); ...
- 概率dp小结
好久之前学过,记得是一次亚洲区的前几天看了看概率dp,然后亚洲区就出了一道概率dp,当时虽然做上了,但是感觉有很多地方没懂,今天起早温习了一下,觉得很多地方茅塞顿开,果然学习的话早上效果最好了. 首先 ...
- 状压DP小结
看了一个多星期状压DP,总算有点明白,大概可以分为两种:数据是在矩阵中的,数据是线性的,在矩阵中的一般就是排兵布阵这一种的,还有一种线性结构中给定条件让你求最大权值,比如求最大权值路线,TSP问题等, ...
- 状态压缩dp小结
最近一段时间算是学了一些状态压缩的题目,在这里做个小结吧 首先是炮兵布阵类题目,这类题目一开始给定一个矩形,要求在上面放置炮兵,如果在一格放了炮兵那么周围的某些格子就不能放炮兵,求最大能放置炮兵的数量 ...
- 数位dp小结
数位dp其实就是一种用来求区间[l, r]满足条件的数的个数.数位是指:个十百千万,而在这里的dp其实相当于暴力枚举每一位数. 我们通过把l, r的每位数分解出来,然后分别求r里满足条件的数有多少,l ...
随机推荐
- WinSock学习笔记
Socket(套接字)◆先看定义: typedef unsigned int u_int; typedef u_int SOCKET; ◆Socket相当于进行网络通信两端的插座,只要对方的Socke ...
- C/C++笔试题(编程题)
面试过程中遇到的编程题整理,于此备录.分享,共勉.(持续更新中......欢迎补充) (1)用户输入M, N值,从1至N开始顺序循环数数,每数到M输出该数值,直至全部输出.写出C程序. 程序代码如下: ...
- 做一次面向对象的体操:将JSON字符串转换为嵌套对象的一种方法
背景与问题 在 <一个略复杂的数据映射聚合例子及代码重构> 一文中,将一个JSON字符串转成了所需要的订单信息Map.尽管做了代码重构和配置化,过程式的代码仍然显得晦涩难懂,并且客户端使用 ...
- 利用可排序Key-Value DB构建时间序列数据库(简论)
为了防止无良网站的爬虫抓取文章,特此标识,转载请注明文章出处.LaplaceDemon/ShiJiaqi. http://www.cnblogs.com/shijiaqi1066/p/5855064. ...
- Linux上查看大文件的开头几行内容以及结尾几行的内容
head -n 50 filePath 查看开头50行的内容 tail -n 50 filePath 查看文件结尾50行的内容
- 转:MD5辅助类
public class MD5Helper { private static MD5 md5 = new MD5CryptoServiceProvider(); private static str ...
- java 数组和集合
1.概念说明 区别:数组固定长度的,集合,数组的长度是可以变化的. List,继承Collection,可重复.有序的对象 Set,继承Collection,不可重复.无序的对象 Map,键值对,提供 ...
- 收音机FM和AM的区别
1.频率区别 FM = Frequency Modulation 调频,微波:微波传输,信号质量高,传输成本低,发射功率小,覆盖范围小,受地理因素影响较大,一般作为城市广播的首选.比如你的家乡城市台, ...
- PLSQL 问题小记
问题1:在一个拼写长字符串的函数中,若出现ora-06502或者ora-06512的错误,则有可能是在拼串的过程中,数据类型的隐式转换出了问题,如:字符类型转为数字,此时会出现错误. 解决方案:在拼串 ...
- Linux(64) 下 Tomcat + java 环境搭建
查看 linux 系统位数 getconf LONG_BIT java JDK下载地址: http://download.oracle.com/otn-pub/java/jdk/8u181-b13/ ...