当前位置:首页 > 题解目录 > 正文内容

【题解】完全背包问题

亿万年的星光3年前 (2023-06-10)题解目录1939

【题目描述】

设有n种物品,每种物品有一个重量及一个价值。但每种物品的数量是无限的,同时有一个背包,最大载重量为M,今从n种物品中选取若干件(同一种物品可以多次选取),使其重量的和小于等于M,而价值的和为最大。

【输入描述】

第一行:两个整数,MM(背包容量,M≤200)和NN(物品数量,N≤30N≤30);

第2…N+1行:每行二个整数Wi,Ci,表示每个物品的重量和价值。

【输出描述】

仅一行,一个数,表示最大总价值。

【样例输入】

10 4 
2 1 
3 3
4 5 
7 9

【样例输出】

12

【题目分析】

1. 状态定义

集合:放入背包的物品方案

限制:选择物品的范围,背包大小

属性:价值

条件:最大

统计量:价值

状态定义:dp[i][j]:在前i个物品中选择物品放入大小为j的背包能获得的最大价值。

初始状态:前0个物品放入大小为j的背包,价值为0。所以dp[0][j] = 0。


2. 状态转移方程

集合:在前i个物品中选择物品放入大小为j的背包的所有方案。

分割集合:是否选择第i物品放入背包


子集1:如果不选择第i物品,那么前i个物品中选择物品放入大小为j的背包,就是在前i-1个物品中选择物品放入大小为j的背包。能获得的最大价值为dp[i][j] = dp[i-1][j]。

子集2:如果选择第i物品,相当于先选了一个第i物品,价值是c[i]。接下来还需要在前i个物品中选择物品放入大小为j-w[i]的背包

。能获得的最大价值为dp[i][j] = dp[i][j-w[i]] + c[i]

以上两种情况求最大值


【参考答案】:二维数组

#include<bits/stdc++.h>
using namespace std;
#define N 35
#define M 205
int m, n, dp[N][M], w[N], c[N];//dp[i][j]:在前i个物品中选择物品放入大小为j的背包能获得的最大价值
int main()
{
    cin >> m >> n;//m:背包容量 n:物品数 
    for(int i = 1; i <= n; ++i)
        cin >> w[i] >> c[i];//w[i]:第i物品的重量 c[i]:第i物品的价值 
    for(int i = 1; i <= n; ++i)
        for(int j = 0; j <= m; ++j)
        {
            if(j >= w[i])
                dp[i][j] = max(dp[i-1][j], dp[i][j-w[i]]+c[i]);
            else
                dp[i][j] = dp[i-1][j];
        }
    cout << "max=" << dp[n][m];
    return 0;
}

【参考答案】:一维数组

#include<bits/stdc++.h>
using namespace std;
#define N 35
#define M 205
int m, n, dp[M], w[N], c[N];//dp[i][j]:在前i个物品中选择物品放入大小为j的背包能获得的最大价值
int main()
{
    cin >> m >> n;//m:背包容量 n:物品数 
    for(int i = 1; i <= n; ++i)
        cin >> w[i] >> c[i];//w[i]:第i物品的重量 c[i]:第i物品的价值 
    for(int i = 1; i <= n; ++i)
        for(int j = w[i]; j <= m; ++j)
            dp[j] = max(dp[j], dp[j-w[i]]+c[i]);
    cout << "max=" << dp[m];
    return 0;
}


    扫描二维码推送至手机访问。

    版权声明:本文由青少年编程知识记录发布,如需转载请注明出处。

    分享给朋友:

    相关文章

    数列分段

    题目描述对于给定的一个长度为N的正整数数列A[i],现要将其分成连续的若干段,并且每段和不超过M(可以等于M),问最少能将其分成多少段使得满足要求。输入格式第1行包含两个正整数N,M,表示了数列A[i...

    【题解】自动晾衣机

    【题目描述】有一个环形可以晾衣服的衣架,有若干个夹子组成,它可以晾不同长度的衣服(占用多个夹子),并且每两件衣服中间要有一个空夹子作为空位,下面需要依次晾干几件长度不一的衣服,请你给出某个夹子的使用情...

    因子分解

    【题目描述】输入一个数,输出其素因子分解表达式。【输入描述】输入一个整数 n (2≤n<100)。【输出描述】输出该整数的因子分解表达式。表达式中各个素数从小到大排列。如果该整数可以分解出因子a...

    【题解】阳光

    【题目描述】给出一个n*n的矩阵,矩阵每个元素数值代表这个位置的阳光情况,给出正整数k,需要我们求出哪一处的k*k 区域的阳光平均值最多,阳光平均值为k*k 区域的阳光总和除于k*k。蒜头君想让我们输...

    【题解】 二维数组转置

    说明输入一个n行m列的数组,输出他的转置,具体来说输出的第i行第j个数字,应是输入的第j行第i个数字。1≤n≤20000;1≤m≤20000;1≤n∗m≤20000;1≤a[i][j]≤1000特别注...

    【题解】搭配购买

    【题目描述】Joe觉得云朵很美,决定去山上的商店买一些云朵。商店里有n朵云,云朵被编号为1,2,…,n,并且每朵云都有一个价值。但是商店老板跟他说,一些云朵要搭配来买才好,所以买一朵云则与这朵云有搭配...