P1439 背包九讲(1):简单的0-1背包

2024-02-18 16:20
文章标签 简单 背包 九讲 p1439

本文主要是介绍P1439 背包九讲(1):简单的0-1背包,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

P1439 背包九讲1:简单的0-1背包

  • 一、原题呈现
    • 1、题目描述
    • 2、输入描述
    • 3、输出描述
    • 4、样例输入
    • 5、样例输出
  • 二、思路分析
    • 这是一个最基础的01背包问题。
  • 三、整体代码

一、原题呈现

1、题目描述

有一个箱子容量为 V(正整数,0<=V<=20000),同时有 n 个物品(0<n<=30),每个物品有一定的体积和价值。要求 n 个物品中,任取若干个装入箱内,在箱子能放得下的前提下,满足箱子内部的价值最大。

2、输入描述

一个整数 v,表示箱子容量
一个整数 n,表示有 n 个物品
接下来 n 个整数,分别表示这 n 个物品的各自体积和价值

3、输出描述

一个整数,表示箱子能装下的最大价值。

4、样例输入

3
2
2 100
4 200

5、样例输出

100

二、思路分析

这是一个最基础的01背包问题。

在这里插入图片描述

这里本人懒得画图,截取了某个大佬的图片来讲解
1、首先我们是按照行来进行遍历。
2、第0行和第0列全都设置为默认0,并且这个单元格的意思是可以选取i行以上的物品,并且背包大小为j,此时能够装到背包内物品价值的最大值
3、我们假设选取第二行第三列的位置,此时我们对于背包的处理有两种情况
(1)不装入物品,那么re[i][j] = re[i - 1][j]
(2)装入该行的物品,那么留给前面的背包质量就为j - weight[i],因此我们可以定位到第i-1行,j - weight[i]列的re值,加上新放入的物品的价值就为re[i][j]的值,即re[i][j] = re[i - 1][j - weight[i]] + value[i]
4、我们将上面两种情况进行求最大值,就是当前re[i][j]的值
5、特殊情况:如果当前背包装不下这个物品(即j - weight[i] < 0),那么我们就直接继承上一行这一列值,即re[i][j] = re[i - 1][j]

三、整体代码

#include<iostream>
#include<math.h>
#include<algorithm> 
using namespace std;int main() {int v, n, i, j;while(cin>>v>>n) {int re[35][21000] = {0};int weight[40], value[40];for(i = 1; i <=n; i++)cin>>weight[i]>>value[i];for(i = 1; i <= n; i++) {for(j = 1; j <= v; j++) {if(j - weight[i] < 0)re[i][j] = re[i - 1][j];else {re[i][j] = max(re[i - 1][j], re[i - 1][j - weight[i]] + value[i]);}}}cout<<re[n][v]<<endl;}return 0;
}

这篇关于P1439 背包九讲(1):简单的0-1背包的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/721736

相关文章

poj2576(二维背包)

题意:n个人分成两组,两组人数只差小于1 , 并且体重只差最小 对于人数要求恰好装满,对于体重要求尽量多,一开始没做出来,看了下解题,按照自己的感觉写,然后a了 状态转移方程:dp[i][j] = max(dp[i][j],dp[i-1][j-c[k]]+c[k]);其中i表示人数,j表示背包容量,k表示输入的体重的 代码如下: #include<iostream>#include<

hdu2159(二维背包)

这是我的第一道二维背包题,没想到自己一下子就A了,但是代码写的比较乱,下面的代码是我有重新修改的 状态转移:dp[i][j] = max(dp[i][j], dp[i-1][j-c[z]]+v[z]); 其中dp[i][j]表示,打了i个怪物,消耗j的耐力值,所得到的最大经验值 代码如下: #include<iostream>#include<algorithm>#include<

csu(背包的变形题)

题目链接 这是一道背包的变形题目。好题呀 题意:给n个怪物,m个人,每个人的魔法消耗和魔法伤害不同,求打死所有怪物所需的魔法 #include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<queue>#include<set>//#include<u>#include<map

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

hdu2289(简单二分)

虽说是简单二分,但是我还是wa死了  题意:已知圆台的体积,求高度 首先要知道圆台体积怎么求:设上下底的半径分别为r1,r2,高为h,V = PI*(r1*r1+r1*r2+r2*r2)*h/3 然后以h进行二分 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#includ

hdu1011(背包树形DP)

没有完全理解这题, m个人,攻打一个map,map的入口是1,在攻打某个结点之前要先攻打其他一个结点 dp[i][j]表示m个人攻打以第i个结点为根节点的子树得到的最优解 状态转移dp[i][ j ] = max(dp[i][j], dp[i][k]+dp[t][j-k]),其中t是i结点的子节点 代码如下: #include<iostream>#include<algorithm

hdu1171(母函数或多重背包)

题意:把物品分成两份,使得价值最接近 可以用背包,或者是母函数来解,母函数(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v) 其中指数为价值,每一项的数目为(该物品数+1)个 代码如下: #include<iostream>#include<algorithm>

usaco 1.3 Prime Cryptarithm(简单哈希表暴搜剪枝)

思路: 1. 用一个 hash[ ] 数组存放输入的数字,令 hash[ tmp ]=1 。 2. 一个自定义函数 check( ) ,检查各位是否为输入的数字。 3. 暴搜。第一行数从 100到999,第二行数从 10到99。 4. 剪枝。 代码: /*ID: who jayLANG: C++TASK: crypt1*/#include<stdio.h>bool h

hdu 2602 and poj 3624(01背包)

01背包的模板题。 hdu2602代码: #include<stdio.h>#include<string.h>const int MaxN = 1001;int max(int a, int b){return a > b ? a : b;}int w[MaxN];int v[MaxN];int dp[MaxN];int main(){int T;int N, V;s

uva 10387 Billiard(简单几何)

题意是一个球从矩形的中点出发,告诉你小球与矩形两条边的碰撞次数与小球回到原点的时间,求小球出发时的角度和小球的速度。 简单的几何问题,小球每与竖边碰撞一次,向右扩展一个相同的矩形;每与横边碰撞一次,向上扩展一个相同的矩形。 可以发现,扩展矩形的路径和在当前矩形中的每一段路径相同,当小球回到出发点时,一条直线的路径刚好经过最后一个扩展矩形的中心点。 最后扩展的路径和横边竖边恰好组成一个直