题解 | 【模板】01背包

【模板】01背包

https://www.nowcoder.com/practice/fd55637d3f24484e96dad9e992d3f62e

解题思路

  1. 问题分析:

    • 第一问:求背包能装下的最大价值(不要求装满)
    • 第二问:求背包恰好装满时的最大价值(要求装满)
  2. 状态定义:

    • :容量为 时能装下的最大价值(不要求装满)
    • :容量为 时恰好装满的最大价值(要求装满)
  3. 初始化:

    • 全部初始化为 (表示空背包,价值为
    • 除了 外,其他都初始化为负无穷(表示还未找到恰好装满的方案)
  4. 状态转移:

对每个物品 : 对每个容量

// 不要求装满

// 要求装满

  1. 结果获取:

    • 第一问:直接输出
    • 第二问:如果 为负无穷,输出 ;否则输出
  2. 优化方案:

    • 使用滚动数组(一维dp)节省空间
    • 从后向前遍历避免重复使用物品
    • 使用 long long 类型避免整数溢出

代码

#include <iostream>
#include <vector>
#include <climits>
using namespace std;

int main() {
    int n, V;
    cin >> n >> V;
    
    // 不要求装满的dp数组
    vector<long long> dp1(V + 1, 0);
    // 恰好装满的dp数组
    vector<long long> dp2(V + 1, LLONG_MIN);
    dp2[0] = 0;
    
    for(int i = 0; i < n; i++) {
        int v, w;
        cin >> v >> w;
        // 从后向前遍历避免重复使用
        for(int j = V; j >= v; j--) {
            if(dp1[j-v] != LLONG_MIN)
                dp1[j] = max(dp1[j], dp1[j-v] + w);
            if(dp2[j-v] != LLONG_MIN)
                dp2[j] = max(dp2[j], dp2[j-v] + w);
        }
    }
    
    cout << dp1[V] << endl;
    cout << (dp2[V] == LLONG_MIN ? 0 : dp2[V]) << endl;
    
    return 0;
}
import java.util.*;

public class Main {
    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        int n = sc.nextInt();
        int V = sc.nextInt();
        
        // 不要求装满的dp数组
        long[] dp1 = new long[V + 1];
        // 恰好装满的dp数组
        long[] dp2 = new long[V + 1];
        Arrays.fill(dp2, Long.MIN_VALUE);
        dp2[0] = 0;
        
        for(int i = 0; i < n; i++) {
            int v = sc.nextInt();
            int w = sc.nextInt();
            // 从后向前遍历避免重复使用
            for(int j = V; j >= v; j--) {
                if(dp1[j-v] != Long.MIN_VALUE)
                    dp1[j] = Math.max(dp1[j], dp1[j-v] + w);
                if(dp2[j-v] != Long.MIN_VALUE)
                    dp2[j] = Math.max(dp2[j], dp2[j-v] + w);
            }
        }
        
        System.out.println(dp1[V]);
        System.out.println(dp2[V] == Long.MIN_VALUE ? 0 : dp2[V]);
        
        sc.close();
    }
}
def solve_knapsack():
    n, V = map(int, input().split())
    
    # 不要求装满的dp数组
    dp1 = [0] * (V + 1)
    # 恰好装满的dp数组
    dp2 = [-float('inf')] * (V + 1)
    dp2[0] = 0
    
    for _ in range(n):
        v, w = map(int, input().split())
        # 从后向前遍历避免重复使用
        for j in range(V, v - 1, -1):
            if dp1[j-v] != -float('inf'):
                dp1[j] = max(dp1[j], dp1[j-v] + w)
            if dp2[j-v] != -float('inf'):
                dp2[j] = max(dp2[j], dp2[j-v] + w)
    
    print(dp1[V])
    print(dp2[V] if dp2[V] != -float('inf') else 0)

if __name__ == "__main__":
    solve_knapsack()

算法及复杂度

  • 算法:动态规划(01背包)
  • 时间复杂度: 是物品数量, 是背包容量
  • 空间复杂度:,使用滚动数组优化
全部评论

相关推荐

来个厂收我吧:首先,市场侧求职我不是很懂。 但是,如果hr把这份简历给我,我会觉得求职人不适合做产品经理。 问题点: 1,简历的字体格式不统一,排版不尽如人意 2,重点不突出,建议参考star法则写个人经历 3,印尼官方货币名称为印度尼西亚卢比(IDR),且GMV690000印尼盾换算为305人民币,总成交额不高。 4,右上角的意向职位在发给其他公司时记得删除。 5,你所有的经历都是新媒体运营,但是你要投市场营销岗位,jd和简历不匹配,建议用AI+提示词,参照多个jd改一下经历内容。 修改建议: 1,统一字体(中文:思源黑体或微软雅黑,英文数字:time new romans),在word中通过表格进行排版(b站学) 2,校招个人经历权重:实习经历=创业经历(大创另算)>项目经历>实训经历>校园经历 3,请将项目经历时间顺序改为倒序,最新的放最上方。 4,求职方向不同,简历文字描述侧重点也需要不同。
点赞 评论 收藏
分享
DKS233:(1)专业技能:Java8也太旧了,最少也要了解到JDK17吧,可以参考现在SpringBoot支持的Java最低版本,熟悉mysql基本理论具体指啥,是锁这种具体原理还是分库分表这些业务场景,spring这些专业词汇,大小写要写对(全篇简历都有这个问题,显得不严谨),熟悉使用框架进行业务开发就别写了,如果要写,起码要写到框架原理部分吧,比如aop,启动原理什么的,springcloud具体指哪些模块呢,写清楚,网关还是鉴权还是什么,“改造”没必要写吧,你直接说用springcloud开发的不就行了(2)项目经历:首先格式就有大问题,时间怎么能换行呢,调整一下,响应速度那个,如果指的是将部分数据从其他数据库转到redis的提升就别写了,因为这个不算难点,redis可以写写分布式这些,比如容灾怎么实现的,数据库同步怎么做的
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务