动态规划(一)

一、背包问题

1.1 01背包问题

**特点:**每件物品最多只用于一次

属性包括:最大值(Max)、最小值(Min)、数量

cpp 复制代码
#include<iostream>
#include<algorithm>

using namespace std;

const int N = 1010;

int n,m;
int v[N],w[N];
int f[N][N];

int main()
{
    cin>>n>>m;

    for(int i = 1;i <= n; i++) cin>>v[i]>>w[i];

    for(int i = 1;i <= n;i ++)
    {
        for(int j = 0;j <= m;j ++)
        {
            f[i][j] = f[i - 1][j];
            if(j >= v[i]) f[i][j] = max(f[i][j],f[i - 1][j - v[i]] + w[i]);
        }
    }
    cout<<f[n][m]<<endl;

    return 0;
}

1.2 完全背包问题

**特点:**每件物品有无限个

cpp 复制代码
#include<iostream>
#include<algorithm>

using namespace std;

const int N = 1010;

int n,m;
int v[N],w[N];
int f[N][N];

int mian()
{
    cin>>n>>m;
    for(int i = 1;i <= n;i ++) cin>>v[i]>>w[i];
    
    for(int i = 1;i <= n;i ++)
        for(int j = 0; j <= m;j ++)
            for(int k = 0;k * v[i] <= j; k ++)
                f[i][j] = max(f[i][j],f[i - 1][j - v[i] * k] + w[i] * k);
    
    cout<<f[n][m]<<endl;
    
    return 0;
}

1.3 多重背包问题

**特点:**每个物品的个数不同

cpp 复制代码
#include<iostream>
#include<algorithm>

using namespace std;

const int N = 25000,M = 2010;

int n,m;
int v[N],w[N];
int f[N];

int main()
{
    cin>>n>>m;
    int cnt = 0;
    for(int i = 1;i <= n;i ++)
    {
        int a,b,s;
        cin>>a>>b>>s;
        int k = 1;
        while(k <= s)
        {
            cnt ++;
            v[cnt] = a * k;
            w[cnt] = b * k;
            s -= k;
            k *= 2;
        }
        if(s > 0)
        {
            cnt ++;
            v[cnt] = a * s;
            w[cnt] = b * s;
        }
    }
    
    n = cnt;
    for(int i = 1;i <= n;i ++)
        for(int j = m;j >= v[i];j --)
           f[j] = max(f[j],f[j - v[i]] + w[i]);
    
    cout<<f[m]<<endl;
    
    return 0;
}

1.4分组背包问题

**特点:**物品有n组,每一组物品中有若干个

cpp 复制代码
#include<iostream>
#include<algorithm>

using namespace std;

const int N = 110;

int n,m;
int v[N][N],w[N][N],s[N];
int f[N];

int main()
{
    cin>>n>>m;

    for(int i = 1;i <= n;i ++)
    {
        cin>>s[i];
        for(int j = 0;j < s[i];j ++)
        {
            cin>>v[i][j]>>w[i][j];
        }
    }


    for(int i = 1;i <= n;i ++)
        for(int j = m;j >= 0;j --)
            for(int k = 0;k < s[i];k ++)
                if(v[i][k] <= j)
                   f[j] = max(f[j],f[j - v[i][k]] + w[i][k]);

    cout<<f[m]<<endl;

    return 0;
}
相关推荐
白熊1882 分钟前
【推荐算法】DeepFM:特征交叉建模的革命性架构
算法·架构·推荐算法
L_cl4 分钟前
【Python 算法零基础 4.排序 ⑪ 十大排序算法总结】
python·算法·排序算法
小刘不想改BUG14 分钟前
LeetCode 70 爬楼梯(Java)
java·算法·leetcode
老歌老听老掉牙23 分钟前
使用 SymPy 进行向量和矩阵的高级操作
python·线性代数·算法·矩阵·sympy
lifallen32 分钟前
Flink checkpoint
java·大数据·算法·flink
比特森林探险记1 小时前
Go 中的 Map 与字符处理指南
c++·算法·golang
安全系统学习2 小时前
网络安全逆向分析之rust逆向技巧
前端·算法·安全·web安全·网络安全·中间件
菜鸟懒懒4 小时前
exp1_code
算法
Winn~4 小时前
JVM垃圾回收器-ZGC
java·jvm·算法
爱coding的橙子4 小时前
每日算法刷题Day24 6.6:leetcode二分答案2道题,用时1h(下次计时20min没写出来直接看题解,节省时间)
java·算法·leetcode