• 背包问题模板


      0/1背包

      01背包是在M件物品取出若干件放在空间为W的背包里,每件物品的体积为W1,W2至Wn,与之相对应的价值为P1,P2至Pn。01背包是背包问题中最简单的问题。01背包的约束条件是给定几种物品,每种物品有且只有一个,并且有权值和体积两个属性。在01背包问题中,因为每种物品只有一个,对于每个物品只需要考虑选与不选两种情况。如果不选择将其放入背包中,则不需要处理。如果选择将其放入背包中,由于不清楚之前放入的物品占据了多大的空间,需要枚举将这个物品放入背包后可能占据背包空间的所有情况。

     1 #include<bits/stdc++.h>
     2 using namespace std;
     3 
     4 const int num = ???;
     5 int t, n, m;// t组数据,n个物品,m背包容量
     6 int v[num], w[num];// wi体积,vi价值
     7 int dp[num];
     8 
     9 int ans(){
    10     //初始化 
    11     for(int i=0; i<num; ++i)    dp[i] = 0; 
    12      
    13     for(int i=1; i<=n; ++i)//第i个物品 
    14         for(int j=m; j>=w[i]; --j)//j为体积 !!倒序 
    15             dp[j] = max(dp[j], dp[j-w[i]]+v[i]);
    16      
    17     return dp[m];
    18 } 
    19 
    20 int main(){
    21     scanf("%d", &t);
    22     while(t--){
    23         scanf("%d %d", &n, &m);
    24         for(int i=1; i<=n; ++i)        scanf("%d", v+i);
    25         for(int i=1; i<=n; ++i)        scanf("%d", w+i);
    26         printf("%d
    ", ans());
    27     }
    28     return 0;
    29 }
    View Code

      完全背包

      有N种物品和一个容量为V的背包,每种物品都有无限件可用。第i种物品的体积是c,价值是w。求解将哪些物品装入背包可使这些物品的体积总和不超过背包容量,且价值总和最大。

     1 #include<bits/stdc++.h>
     2 using namespace std;
     3 
     4 const int num = ???;
     5 int t, n, m;
     6 int w[num], v[num];
     7 int dp[num];
     8 
     9 int fullans(){
    10     //初始化 
    11     for(int i=0; i<num; ++i)    dp[i] = 0; 
    12      
    13     for(int i=1; i<=n; ++i)//第i个物品 
    14         for(int j=w[i]; j<=m; ++j)//j为体积 !!顺序 
    15             dp[j] = max(dp[j], dp[j-w[i]]+v[i]);
    16      
    17     return dp[m];    
    18 }
    19 
    20 int main(){
    21     scanf("%d", &t);
    22     while(t--){
    23         scanf("%d %d", &n, &m);
    24         for(int i=1; i<=n; ++i)        scanf("%d", w+i);
    25         for(int i=1; i<=n; ++i)        scanf("%d", v+i);
    26         printf("%d
    ",fullans());
    27     }
    28     return 0;
    29 }
    View Code

      多重背包

      有N种物品和一个容量为V的背包。第i种物品最多有n[i]件可用,每件费用是c[i],价值是w[i]。求解将哪些物品装入背包可使这些物品的费用总和不超过背包容量,且价值总和最大。

     1 d#include<bits/stdc++.h>
     2 using namespace std;
     3 
     4 const int N = ???;
     5 int V, m;
     6 int dp[N];
     7 int w[N], num[N];
     8 
     9 void mul(int n,int w){
    10     if(n*w >= V){
    11         for(int i=w; i<=V; i++)
    12             dp[i] = max(dp[i], dp[i-w]+w);
    13         return ;
    14     }
    15     int k = 1;
    16     int nC = n;
    17     while(k < nC){//01背包,做简单的优化
    18         for(int i=V; i>=k*w; i--)
    19             dp[i] = max(dp[i], dp[i-k*w]+k*w);
    20         nC -= k;
    21         k *= 2;
    22     }
    23     for(int i=V; i>=nC*w; i--)
    24         dp[i] = max(dp[i], dp[i-nC*w]+nC*w);
    25 }
    26 int main(){
    27     while(scanf("%d %d", &m, &V) && m && V){
    28         for(int i=0; i<=V; ++i)        dp[i] = 0;
    29         for(int i=0; i<m; ++i)        scanf("%d", w+i);
    30         for(int i=0; i<m; ++i)        scanf("%d", num+i);
    31 
    32     //多重背包    
    33         for(int i=0; i<m; ++i)
    34             mul(num[i], w[i]);
    35 
    36         printf("%d
    ",dp[V]);
    37     }
    38     return 0;
    39 }
    View Code
  • 相关阅读:
    第十五周总结
    第二阶段冲刺
    课堂练习
    第十四周总结
    WP8 NavigationInTransition实现页面切换效果
    WP8学习笔记:如何在页面显示前自动转向到其他页面
    WP8简单的计算器
    WP8滑动条(Slider)控件的使用
    Wp8滚动区域(ScrollViewer)控件的使用
    C#中ToString格式大全
  • 原文地址:https://www.cnblogs.com/0424lrn/p/12262880.html
Copyright © 2020-2023  润新知