当前位置:   article > 正文

动态规划:完全背包问题_完全背包问题 动态规划

完全背包问题 动态规划

ACwing #3. 完全背包问题

完全背包问题和01背包问题很相似。
01背包问题每个物品只能选一个,而完全背包问题每个物品可以选无限次。

DP问题的关键是找到状态转移方程

①定义f[i][j]表示从前 i 个物品中选择,体积为 j 的时候的最大价值。
②那么转移方程f[i][j] = max(f[i - 1][j],f[i - 1][j - v[i]],f[i - 1][j - 2 * v[i]],.....,f[i - 1][j - k * v[i]],....)

因此代码就是:

  1. #include<iostream>
  2. using namespace std;
  3. const int N = 1010;
  4. int f[N][N];
  5. int v[N],w[N];
  6. int main()
  7. {
  8. int n,m;
  9. cin>>n>>m;
  10. for(int i = 1 ; i <= n ;i ++)
  11. {
  12. scanf("%d%d",&v[i],&w[i]);
  13. }
  14. for(int i = 1 ; i<=n ;i++)
  15. for(int j = 1 ; j<=m ;j++)
  16. {
  17. for(int k = 0 ; k*v[i]<=j ; k++)
  18. f[i][j] = max(f[i][j],f[i-1][j-k*v[i]]+k*w[i]);
  19. }
  20. printf("%d",f[n][m]
  21. return 0;
  22. }

由于数据量级的原因,此代码肯定会发生TLE,因此需要进行优化。

f[i , j ] = max( f[i-1,j] , f[i- 1,j - v[i]+ w[i] ,  f[i - 1,j-2 * v[i]]+2 * w[i] , f[i -1,j - 3 * v[i]]+3 * w[i] , .....)
f[i , j - v[i]]= max(       f[i - 1,j - v[i]] ,         f[i - 1,j - 2 * v[i]] + w[i] , f[i - 1,j- 3 * v[i]]+2 * w[i] , .....)

由上两式,可得出如下递推关系: 
                        f[i][j]=max(f[i, j - v[i] ] + w[i] , f[i - 1][j])  

因此优化后的代码变为:

  1. #include <iostream>
  2. using namespace std;
  3. const int N=1010;
  4. int v[N],w[N],f[N][N];
  5. int n,m;
  6. int main()
  7. {
  8. scanf("%d%d",&n,&m);
  9. for(int i = 1;i <= n;i++){
  10. scanf("%d%d",&v[i],&w[i]);
  11. }
  12. for(int i = 1;i <= n;i++)
  13. for(int j = 1;j <= m;j++){
  14. f[i][j] = f[i - 1][j];
  15. if(j >= v[i])
  16. f[i][j] = max(f[i][j],f[i][j - v[i]] + w[i]);
  17. }
  18. printf("%d",f[n][m]);
  19. return 0;
  20. }

        可以看出代码与01背包非常相似,因此尝试能否做进一步优化。

优化完成后的状态转移方程:f[j] = max(f[j], f[j - v[i]] + w[i]) 

代码如下:

  1. #include <iostream>
  2. using namespace std;
  3. const int N=1010;
  4. int v[N],w[N],f[N];
  5. int n,m;
  6. int main()
  7. {
  8. scanf("%d%d",&n,&m);
  9. for(int i = 1;i <= n;i++){
  10. scanf("%d%d",&v[i],&w[i]);
  11. }
  12. for(int i=1;i<=n;i++)
  13. for(int j=v[i];j<=m;j++){
  14. f[j]=max(f[j],f[j-v[i]]+w[i]);
  15. }
  16. printf("%d",f[m]);
  17. return 0;
  18. }
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/小小林熬夜学编程/article/detail/432262
推荐阅读
相关标签
  

闽ICP备14008679号