DP 要求将承载量花光的01背包问题

前言:很久没有发博客了,以后会捡起来,之后很长一段时间内我都会把精力放在攻克DP问题上,所以会经常上传一些DP学习笔记,把一些比较好的,没见过类型的DP问题都会传上来,希望能够变强吧。

因为今天很清醒的意识到世上有很多很优秀很惊艳的人,自己也要努力,努力成为那样的人啊 。

例题 洛谷P1164 小A点菜

分析:乍看就是一个普通的01背包,仔细看下要求钱是要全花完的,而且dp数组内保存的不是最大价值,而应该是方法的数量,这样处理起来是很不一样了

这道题第三次做,之前两次做都是最后无奈的屈从了题解,今天稍微看了看,就很轻松的做出来了,还是有进步的嘛。

在状态转移方程上,因为不再要求最大价值了,而且要把钱花光,所以01背包的那些max都不虚要了,用dp[i][j]表示前i种物品花j元的话,状态转移方程就是

dp[i][j]=dp[i-1][j]+dp[i-1][j-a[i]] (a[i]是一个菜花的钱数)

初始化一下:dp[0][0]=1 

然后加个滚动数组就可以了。

上代码:

 1 #include<bits/stdc++.h>
 2 using namespace std;
 3 typedef long long ll;
 4 const double pi=acos(-1);
 5 int a[110];
 6 int dp[10010];
 7 int main(){
 8     int n,m;scanf("%d%d",&n,&m);
 9     for(int i=0;i<n;i++)scanf("%d",&a[i]);
10     dp[0]=1;
11     for(int i=0;i<n;i++){
12         for(int j=m;j>=a[i];j--)
13         dp[j]=dp[j]+dp[j-a[i]];
14     }
15     cout<<dp[m]<<endl; 
16     return 0;
17 }
原文地址:https://www.cnblogs.com/qingjiuling/p/10138117.html