【算法-动态规划】0-1 背包问题

news2024/10/6 6:01:55

💝💝💝欢迎来到我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。
img

  • 推荐:kuan 的首页,持续学习,不断总结,共同进步,活到老学到老
  • 导航
    • 檀越剑指大厂系列:全面总结 java 核心技术点,如集合,jvm,并发编程 redis,kafka,Spring,微服务,Netty 等
    • 常用开发工具系列:罗列常用的开发工具,如 IDEA,Mac,Alfred,electerm,Git,typora,apifox 等
    • 数据库系列:详细总结了常用数据库 mysql 技术点,以及工作中遇到的 mysql 问题等
    • 懒人运维系列:总结好用的命令,解放双手不香吗?能用一个命令完成绝不用两个操作
    • 数据结构与算法系列:总结数据结构和算法,不同类型针对性训练,提升编程思维,剑指大厂

非常期待和您一起在这个小小的网络世界里共同探索、学习和成长。💝💝💝 ✨✨ 欢迎订阅本专栏 ✨✨

博客目录

        • 1.问题描述
        • 2.问题解析
        • 3.二维解答
        • 4.一维解答
        • 5.继续优化

1.问题描述

0-1 背包问题(0-1 Knapsack Problem)是一个经典的组合优化问题,通常出现在计算机科学和数学中。它的背景是在给定一组物品,每个物品有一个特定的重量和价值,以及一个固定容量的背包。问题的目标是确定应该选择哪些物品放入背包,以使得它们的总重量不超过背包的容量,并且它们的总价值最大化。

0-1 背包问题之所以称为 0-1,是因为对于每个物品,你只有两种选择:要么将它放入背包(表示为 1),要么不放入背包(表示为 0)。不能将一个物品部分放入背包。

2.问题解析
/*
  1. n个物品都是固体,有重量和价值
  2. 现在你要取走不超过 10克 的物品
  3. 每次可以不拿或全拿,问最高价值是多少
      编号 重量(g)  价值(元)                        简称
      1   4       1600           黄金一块   400     A
      2   8       2400           红宝石一粒 300     R
      3   5       30             白银一块          S
      4   1       1_000_000      钻石一粒          D
  1_001_630 贪心解
  1_002_400 正确解
*/
/*
      0   1   2   3   4   5   6   7   8   9   10
  0   0   0   0   0   A   A   A   A   A   A   A       黄金
  1   0   0   0   0   A   A   A   A   R   R   R       红宝石
  2   0   0   0   0   A   A   A   A   R   R   R       白银
  3   0   D   D   D   D   DA  DA  DA  DA  DR  DR      钻石
  if(装不下) {
      dp[i][j] = dp[i-1][j]
  } else { 装得下
      dp[i][j] = max(dp[i-1][j], item.value + dp[i-1][j-item.weight])
  }
*/
3.二维解答
static class Item {
    int index;
    String name;
    int weight;
    int value;

    public Item(int index, String name, int weight, int value) {
        this.index = index;
        this.name = name;
        this.weight = weight;
        this.value = value;
    }

    @Override
    public String toString() {
        return "Item(" + name + ")";
    }
}

public static void main(String[] args) {
    Item[] items = new Item[]{
            new Item(1, "黄金", 4, 1600),
            new Item(2, "宝石", 8, 2400),
            new Item(3, "白银", 5, 30),
            new Item(4, "钻石", 1, 10_000),
    };
    System.out.println(select(items, 10));
}

static int select(Item[] items, int total) {
    int[][] dp = new int[items.length][total + 1];
    print(dp);
    final Item item0 = items[0];
    for (int j = 0; j < total + 1; j++) {
        if (j >= item0.weight) {//装得下
            dp[0][j] = item0.value;
        }
    }
    print(dp);
    for (int i = 1; i < items.length; i++) {
        final Item item = items[i];
        for (int j = 0; j < total + 1; j++) {
            if (j >= item.weight) {//装得下
                dp[i][j] = Integer.max(dp[i - 1][j], item.value + dp[i - 1][j - item.weight]);
            } else {
                dp[i][j] = dp[i - 1][j];
            }
        }
        print(dp);
    }
    return dp[items.length - 1][total];
}

static void print(int[][] dp) {
    System.out.println(StringUtil.repeat("   " + "-", 20));
    Object[] array = IntStream.range(0, dp[0].length + 1).boxed().toArray();
    System.out.printf((StringUtil.repeat("%5d ", dp[0].length)) + "%n", array);
    for (int[] d : dp) {
        array = Arrays.stream(d).boxed().toArray();
        System.out.printf((StringUtil.repeat("%5d ", d.length)) + "%n", array);
    }
}
4.一维解答
static class Item {
    int index;
    String name;
    int weight;
    int value;

    public Item(int index, String name, int weight, int value) {
        this.index = index;
        this.name = name;
        this.weight = weight;
        this.value = value;
    }

    @Override
    public String toString() {
        return "Item(" + name + ")";
    }
}

public static void main(String[] args) {
    Item[] items = new Item[]{
            new Item(1, "黄金", 4, 1600),
            new Item(2, "宝石", 8, 2400),
            new Item(3, "白银", 5, 30),
            new Item(4, "钻石", 1, 10_000),
    };
    System.out.println(select(items, 10));
}

static int select(Item[] items, int total) {
    int[] dp = new int[total + 1];
    System.out.println(Arrays.toString(dp));
    final Item item0 = items[0];
    for (int j = 0; j < total + 1; j++) {
        if (j >= item0.weight) {//装得下
            dp[j] = item0.value;
        }
    }
    System.out.println(Arrays.toString(dp));
    for (int i = 1; i < items.length; i++) {
        final Item item = items[i];
        for (int j = total; j > 0; j--) {
            if (j >= item.weight) {//装得下
                dp[j] = Integer.max(dp[j], item.value + dp[j - item.weight]);
            }
        }
        System.out.println(Arrays.toString(dp));
    }
    return dp[total];
}
5.继续优化
static class Item {
    int index;
    String name;
    int weight;
    int value;

    public Item(int index, String name, int weight, int value) {
        this.index = index;
        this.name = name;
        this.weight = weight;
        this.value = value;
    }

    @Override
    public String toString() {
        return "Item(" + name + ")";
    }
}

public static void main(String[] args) {
    Item[] items = new Item[]{
            new Item(1, "黄金", 4, 1600),
            new Item(2, "宝石", 8, 2400),
            new Item(3, "白银", 5, 30),
            new Item(4, "钻石", 1, 10_000),
    };
    System.out.println(select(items, 10));
}

static int select(Item[] items, int total) {
    int[] dp = new int[total + 1];
    for (int i = 0; i < items.length; i++) {
        final Item item = items[i];
        for (int j = total; j > 0; j--) {
            if (j >= item.weight) {//装得下
                dp[j] = Integer.max(dp[j], item.value + dp[j - item.weight]);
            }
        }
        System.out.println(Arrays.toString(dp));
    }
    return dp[total];
}

注意:内层循环需要倒序,否则 dp[j - item.weight] 的结果会被提前覆盖

觉得有用的话点个赞 👍🏻 呗。
❤️❤️❤️本人水平有限,如有纰漏,欢迎各位大佬评论批评指正!😄😄😄

💘💘💘如果觉得这篇文对你有帮助的话,也请给个点赞、收藏下吧,非常感谢!👍 👍 👍

🔥🔥🔥Stay Hungry Stay Foolish 道阻且长,行则将至,让我们一起加油吧!🌙🌙🌙

img

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1085417.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

解锁 OpenAI 密钥的新利器:OpenAI Key 有效性查询工具,支持GPT4

在寻找新的创意或解决问题时&#xff0c;OpenAI 的 GPT 模型是一个强大的工具。但是&#xff0c;一旦您拥有了自己的 OpenAI 密钥&#xff0c;您可能会遇到一个常见的问题 - 如何验证您的密钥是否有效&#xff1f;这个问题可能令人困扰&#xff0c;但不必再担心了。现在&#x…

多功能按键中断

key1 开关实现led1亮灭,key2开关实现蜂鸣器开关,key3开关实现风扇开关 main.c #include "uart.h" #include "key_it.h" #include "led.h" int main() {char c;char *s;uart4_init();//串口初始化all_led_init();key_it_config();fengshan_init…

Day 05 python学习笔记

循环 应用&#xff1a;循环轮播图 最基础、最核心 循环&#xff1a;周而复始&#xff0c;谓之循环 (为了代码尽量不要重复) while循环 while的格式 索引定义 while 表达式&#xff08;只要结果为布尔值即可&#xff09;&#xff1a; 循环体 通过条件的不断变化&#xff0c;从…

1315. 网格 - 卡特兰数

1315. 网格 - AcWing题库 只要是触及上面这条红线的&#xff0c;就以第一次触及的点为起点沿红线反转&#xff0c;终点的位置与红线对称的位置可以看作触及红线的路线的终点。 yx1 横坐标容易得出时m - 1&#xff0c;&#xff08;m n&#xff09; - (m - 1)得出纵坐标n 1 …

JVM面试题:(四)四种引用方式强弱软虚

四种引用方式&#xff1a; 强引用 强引用是平常中使用最多的引用&#xff0c;强引用在程序内存不足&#xff08;OOM&#xff09;的时候也不会被回收&#xff0c;使用 方式&#xff1a; String str new String(“str”); System.out.println(str); 软引用 软引用在程序内存不…

07-网络篇-抓包分析TCP

为了抓包方便一些&#xff0c;我在ubuntu虚拟机运行服务端程序&#xff0c;而在windows运行客户端程序&#xff0c;关于客户端与服务端程序如下。 ##1.程序 客户端&#xff1a; vs_client.cpp #include "stdafx.h" #include <iostream> #include <winsock2…

MapStruct_概念、如何使用、子集和映射、合并、Spring方式、表达式、自定义切面处理

文章目录 ①. 什么是MapStruct&#xff1f;②. 如何使用MapStruct?③. 子集和映射④. 合并映射⑤. Spring依赖注入⑥. 常量、默认值和表达式⑦. 自定义切面处理 ①. 什么是MapStruct&#xff1f; ①. MapStruct是一款基于Java注解的对象属性映射工具,使用的时候我们只要在接口…

SQL 教程||SQL 简介

SQL 简介&#xff08;开个新坑&#xff09; SQL&#xff08;结构化查询语言&#xff09;是用于访问和操作数据库中的数据的标准数据库编程语言。 SQL是关系数据库系统的标准语言。所有关系数据库管理系统(RDMS)&#xff0c;如MySQL、MS Access、Oracle、Sybase、Informix、Po…

第四章——密码学的数学引论

一.数论 1.素数 200以内的素数&#xff1a; 2 3 5 7 11 13 17 19 23 29 31 37 41 43 47 53 59 61 67 71 73 79 83 89 97 101 103 107 109 113 127 131 137 139 149 151 157 163 167 173 179 181 191 193 197 199 算术基本定理&#xff1a; 任何一个不等于0的正整数a都可以写…

接上面 通过索引index 和类型Java.lang.string.来注入

index 属性表示构造函数参数的索引&#xff0c;value 属性表示构造函数参数的值。

【web实现右侧弹窗】JS+CSS如何实现右侧缓慢弹窗动态效果『附完整源码下载』

文章目录 写在前面涉及知识点页面效果1、页面DOM创建1.1创建底层操作dom节点1.2 创建存放弹窗dom节点 2、页面联动功能实现&#xff08;关闭与弹出&#xff09;2.1 点击非右侧区域实现关闭2.2 点击叉叉及关闭按钮实现关闭功能 3、完整源码包下载3.1百度网盘3.2 123云盘3.3邮箱留…

没用的知识增加了,尝试用文心实现褒义词贬义词快速分类

尝试用文心实现褒义词贬义词快速分类 一、我的需求二、项目环境搭建千帆SDK安装及使用流程 三、项目实现过程创建应用获取签名调用接口计算向量积总结 百度世界大会将于10月17日在北京首钢园举办&#xff0c;今天进入倒计时五天了。通过官方渠道的信息了解到&#xff0c;这次是…

【APUE】文件系统 — 进程环境

目录 一、再提 main 函数 二、进程的终止 2.1 正常终止 2.1.1 从 main 函数用 return 返回 2.1.2 主动调用 exit 函数 2.1.3 钩子函数 2.1.4 调用 _exit 或 _Exit 2.2 异常终止 三、命令行参数的分析 3.1 getopt 3.2 示例 四、环境变量 4.1 简介 4.2 查看环境…

内网收集哈希传递

1.内网收集的前提 获得一个主机权限 补丁提权 可以使用 systeminfo 然后使用python脚本找到缺少的补丁 下载下来 让后使用exp提权 收集信息 路由信息 补丁接口 dns域看一看是不是域控 扫描别的端口 看看有没有内在的web网站 哈希传递 哈希是啥 哈希…

墨者学院 WordPress 远程命令执行漏洞(CVE-2018-15877)

1. 背景介绍 近日&#xff0c;WordPress 插件Plainview Activity Monitor被曝出存在一个远程命令执行漏洞。Plainview Activity Monitor 是一款网站用户活动监控插件。 远程攻击者可以通过构造的url来诱导wordpress管理员来点击恶意链接最终导致远程命令执行 2.影响范围 Pla…

ajax同步与异步,json-serve的安装与使用,node.js的下载

20.ajax json 轻量级的数据格式做配置文件网络传输 xml 重量级的数据格式 可扩展标记语言做配置文件网络传输 现在目前主流就是大量采用json做网络传输数据格式 1.ajax的概念: 与服务器进行’通信’的一种技术,能够实现异步的刷新页面 **同步:**按照顺序一步步的执行,容易造…

Python应用-矩阵乘法-特征提取

目录 常规运算应用场景&#xff1a;特征提取特征矩阵权重矩阵举例说明代码展示 常规运算 import numpy as npmatrix1 np.array([[1, 2], [3, 4]]) matrix2 np.array([[5, 6], [7, 8]]) result np.dot(matrix1, matrix2) print(result) 输出结果&#xff1a; [[19 22][43 …

是谁还没听过杨氏矩阵~原理和实现代码都已经准备好了

有一个数字矩阵&#xff0c;矩阵的每行从左到右是递增的&#xff0c;矩阵从上到下是递增的&#xff0c;请编写程序在这样的矩阵中查找某个数字是否存在。 要求&#xff1a;时间复杂度小于O(N); 看到这个题目时&#xff0c;我们会马上想到暴力求解&#xff0c;即遍历这个矩阵的每…

Excel往Word复制表格时删除空格

1.背景 在编写文档&#xff0c;经常需要从Excel往Word里复制表格 但是复制过去的表格前面会出现空格&#xff08;缩进&#xff09; 再WPS中试了很多方法&#xff0c;终于摆脱了挨个删除的困扰 2. WPS排版中删除 选择表格菜单栏-选在【开始】-【排版】选择【更多段落处理】-【段…

论文阅读/写作扫盲

第一节&#xff1a;期刊科普 JCR分区和中科院分区是用于对期刊进行分类和评估的两种常见方法。它们的存在是为了帮助学术界和研究人员更好地了解期刊的学术质量、影响力和地位。 JCR分区&#xff08;Journal Citation Reports&#xff09;&#xff1a;JCR分区是由Clarivate Ana…