Basics on Dynamic Programming(DP) 9 - Backpack

backpack 背包问题

1. 0-1背包问题

vector<int> dp(M+1, 0);  // 手里奖券数,能买的最大的价值
// 在执行i次循环后(此时已经处理i个物品),前i个物体放到容量v时的最大价值,即之前的f[i][v]
for(int i=0; i<N; i++) {
  for(int j=M; j>=need[i]; j--)  // 现在有的钱
    dp[j] = max(dp[j], dp[j-need[i]]+val[i]);  // 不选,或选的最大值
}
cout << dp[M];

2. 硬币找零问题

人民币的硬币系统是 100,50,20,10,5,2,1,0.5,0.2,0.1,0.05,0.02,0.01 元,采用这些硬币我们可以对任何一个工资数用贪心算法求出其最少硬币数。 
但不幸的是:我们可能没有这样一种好的硬币系统,因此用贪心算法不能求出最少的硬币数,甚至有些金钱总数还不能用这些硬币找零。
例如,如果硬币系统是 40,30,25 元,那么 37元就不能用这些硬币找零;95元的最少找零硬币数是 3。
又如,硬币系统是 10,7,5,1元,那么 12 元用贪心法得到的硬币数为 3,而最少硬币数是 2。

你的任务就是:对于任意的硬币系统和一个金钱数,请你编程求出最少的找零硬币数;
如果不能用这些硬币找零,请给出一种找零方法,使剩下的钱最少(剩下钱数最少的找零方案中的最少硬币数。)。 
// DP[i]表示i最少需要的硬币个数,假设给定的硬币面值为a,b,c,d……
// 则dp[i]=min(dp[i-a],dp[i-b],dp[i-c],dp[i-d],……)+1 (i-a,i-b,……>=0)
int Solve(vector<int> &num, int money) {
  int n = num.size();
  // 初始化dp
  vector<int> dp(money + 1, 100001);  // 初始化为题目给定的最大值
  dp[0] = 0;
  // for (auto i : num) dp[i] = 1;  // 需要1个硬币 这个可以不调用,dp[0]就可以了

  sort(num.begin(), num.end());  // 优化算法,从小到大排序

  for (int i = 1; i <= money; i++) {              // 总钱数,开始循环
    for (int j = 0; num[j] <= i && j < n; j++) {  // 注意终止条件num[j] <= i
      dp[i] = min(dp[i], dp[i - num[j]] + 1);
    }
  }

  while (dp[money] == 100001)  //所给数据不能用硬币表示
    money--;
  return dp[money];
}

3.Number of Ways to Represent N Cents

Given an infinite number of quarters (25 cents), dimes (10 cents), nickels (5 cents) and pennies (1 cent), write code to calculate the number of ways of representing n cents.
Example
n = 11
11 = 1 + 1 + 1... + 1
   = 1 + 1 + 1 + 1 + 1 + 1 + 5
   = 1 + 5 + 5
   = 1 + 10
return 4
// 错误解法
int waysNCents(int n) {
  int coins[4] = {1, 5, 10, 25};
  vector<int> dp(n + 1, 0);
  dp[0] = 1;  // 注意初始化!
  for (int i = 1; i <= n; i++) {
    for (int j = 0; coins[j] <= i && j < 4; j++) {
      dp[i] += dp[i - coins[j]];  // dp[6] = dp[1] + dp[5],1,5和5,1重复了
    }
  }
  return dp[n];
}
// 正确解法!!
int waysNCents(int n) {
  int coins[4] = {1, 5, 10, 25};
  vector<int> dp(n + 1, 0);
  dp[0] = 1;  // 注意初始化!
  // Pick all coins one by one and update the table[] values
  // after the index greater than or equal to the value of the
  // picked coin
  for (int i = 0; i < 4; i++) {  // 区别:外层循环是coins
    for (int j = coins[i]; j <= n; j++) {
      dp[j] += dp[j - coins[i]];  // 肯定选了这个coin
    }
  }
  return dp[n];
}
Loading Disqus comments...
Table of Contents