动态规划(一)

一、背包问题

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;
}
相关推荐
Z1Jxxx19 小时前
01序列01序列
开发语言·c++·算法
汽车仪器仪表相关领域20 小时前
全自动化精准检测,赋能高效年检——NHD-6108全自动远、近光检测仪项目实战分享
大数据·人工智能·功能测试·算法·安全·自动化·压力测试
Doro再努力21 小时前
【数据结构08】队列实现及练习
数据结构·算法
清铎1 天前
leetcode_day12_滑动窗口_《绝境求生》
python·算法·leetcode·动态规划
linweidong1 天前
嵌入式电机:如何在低速和高负载状态下保持FOC(Field-Oriented Control)算法的电流控制稳定?
stm32·单片机·算法
net3m331 天前
单片机屏幕多级菜单系统之当前屏幕号+屏幕菜单当前深度 机制
c语言·c++·算法
mmz12071 天前
二分查找(c++)
开发语言·c++·算法
Insight1 天前
拒绝手动 Copy!一文吃透 PyTorch/NumPy 中的广播机制 (Broadcasting)
算法
CoovallyAIHub1 天前
工业视觉检测:多模态大模型的诱惑
深度学习·算法·计算机视觉