题目来源:点击打开链接
题目内容:
题目描述
有一个箱子容量为V(正整数,0<=V<=20000),同时有n个物品(0<n<=30,每个物品有一个体积(正整数)。
要求n个物品中,任取若干个装入箱内,使箱子的剩余空间为最小。
输入输出格式
输入格式:
一个整数,表示箱子容量
一个整数,表示有n个物品
接下来n行,分别表示这n 个物品的各自体积
输出格式:
一个整数,表示箱子剩余空间。
样例:
输入
24
6
8
3
12
7
9
7
输出:
0
思路:
非常标准的动态规划01背包问题,将它的重量看作它的价值,求剩余最小量就是能装的最大量
求n个物品中,任取若干个装入箱内,总价值最大。
对于每一个物体,都有两种状态:装 与 不装
那么,对于任意重量m的最大价值 dp[m] = max(dp[ m - w[i] ] + w[i],dp[m] )(w为重量(即价值))
其中,dp [ m - w[i] ] 指在装了物品i后,箱子的剩余容量能装的最大重量
dp ( m - w[i] ) + w[i] 指在在装了物品i后,箱子能装的最大重量
dp[m]表示当前的价值
max(dp[m],dp[m-w[i]]+w[i]);表示判断加上物体价值大还是不加物体价值大,保留价值大的那一个。
见代码
#include<iostream>
using namespace std;
int dp[200010];//箱子的重量
int w[40];//每件物体的重量
int main()
{
int sum;
cin>>sum;//箱子的重量
int n;
cin>>n;
for(int i=1;i<=n;++i)
cin>>w[i];
if(!sum)//判断箱子的重量是否为零
{
cout<<sum;
return 0;
}
for(int i=1;i<=n;++i)
for(int j=sum;j>=w[i];--j)//一定是从j=sum开始,不然会出现错误
dp[j]=max(dp[j],dp[j-w[i]]+w[i]);//比较大小
cout<<sum-dp[sum];//dp[sum]表示的是装的最大重量。求的是剩余的重量
return 0;
}