Charm Bracelet —— 0-1背包

题意:
有n个重量和价值分别为wi,vi的物品。从这些物品中挑选出总重量不超过C的物品,求所有挑选方案中价值总和的最大值。

输入:

n = 4, C = 5
(w,v) = {(2,3),(1,2),(3,4),(2,2)}

输出:

7 (选择第0,1,3号物品)


典型的0-1背包问题

f[j]:背包还能装的重量为 j 时的最大价值:

#include<stdio.h>
#include<string.h>
int maxn(int i,int j) {return i > j ? i : j;}
int main()
{
    int n,C,W,V,f[13000];

    scanf("%d%d", &n,&C);
    memset(f, 0, sizeof(f));
    for(int i=0; i<n; i++)
    {
        scanf("%d%d", &W,&V);
        for(int j=C; j>=W; j--)
            f[j] = maxn(f[j],f[j-W]+V);
    }
    printf("%d\n", f[C]);
    return 0;
}

f[j]:背包已放的重量为 j 时的最大价值:

#include<stdio.h>
#include<string.h>
#include<algorithm>
using namespace std;
int main()
{
    int n,C,W,V,f[13000];

    scanf("%d%d", &n,&C);
    memset(f, 0, sizeof(f));
    for(int i=0; i<n; i++)
    {
        scanf("%d%d", &W,&V);
        for(int j=C-W; j>=0; j--)
            f[j+W] = max(f[j+W],f[j]+V);
    }
    printf("%d\n", f[C]);
    return 0;
}
dp可以算出每个时期的最大值,最后输出 f[k] 时就代表最大容量为 k 时的最大价值。

 

这里用二维数组的话,数组大小开不下,所以只能用滚动数组,不过也贴下代码:

#include<stdio.h>
int d[500][500];
int maxn(int i,int j) {return i > j ? i : j;}
int main()
{
    int n,C,i,j,w[3500],v[3500];

    scanf("%d%d", &n,&C);
    for(i=1; i<=n; i++)
        scanf("%d%d", w+i,v+i);
    for(i=n; i>=1; i--) //必须从n到1循环(由于是通过递归得出答案的!!)
    {
        for(j=0; j<=C; j++)
        {
            d[i][j] = (i == n ? 0 : d[i+1][j]);
            if(j >= w[i]) d[i][j] = maxn(d[i+1][j],d[i+1][j-w[i]]+v[i]);
        }
    }
    printf("%d\n", d[1][C]);
    return 0;
}

上面的程序提交会是 runtime error ,但也是一种方式,可能更好理解一点;

也可以考虑用记忆化搜索的思路:

#include<stdio.h>
#include<string.h>
#include<algorithm>
using namespace std;
int dp[500][500]; //记忆化数组
int n, w[3500], v[3500], C;

int dfs(int i, int j)
{
    if(dp[i][j] >= 0) return dp[i][j]; //已经计算过的话直接使用之前的结果
    int res;
    if(i == n) res = 0; //没有下一个选项了就返回0
    else if(j < w[i]) res = dfs(i+1,j); //如果剩余容量小于第i个物体的重量,就尝试下一个
    else res = max(dfs(i+1,j), dfs(i+1,j-w[i])+v[i]); //取或不取选最大方案
    return dp[i][j] = res; //将结果记录在数组中
}

int main()
{
    scanf("%d%d", &n,&C);
    for(int i=0; i<n; i++)
        scanf("%d%d", w+i,v+i);
    memset(dp, -1, sizeof(dp));
    printf("%d\n", dfs(0,C));
    return 0;
}

for循环的dp形式就是利用递推关系,递推、递推、还是递推!!

你可能感兴趣的:(动态规划,记忆化搜索,0-1背包)