代码随想录打卡—day42—【DP】— 8.27 01背包基础

news2025/1/12 13:20:52

1 01背包基础

背包概述:

1.1 01背包是什么

有n件物品和一个最多能背重量为w 的背包。第i件物品的重量是weight[i],得到的价值是value[i] 。每件物品只能用一次,求解将哪些物品装入背包里物品价值总和最大。

1.2 01背包二维数组

二维数组还比较好理解,五步法详见代码注释,AC代码:

#include<iostream>
using namespace std;

const int N = 1e3 + 10;

int weight[N];
int val[N];
int f[N][N];  // (i,j)表示只选取前i个商品 容积j的背包能够装的最大价值
/*

if(j < weight[i])f[i][j] = f[i-1][j];  //装不下就只有不装一条路
else f[i][j] = max(f[i-1][j],f[i-1][j-weight[i]]+val[i]);  //装的下就有装和不装两种选择
              不装i       装i

第0行--f[0][j]
for(int j = 0; j <= m ; j++)
{
    if(j >= weight[0])f[0][j] = val[0];
    else f[0][j] = 0;
}
第0列—— f[i][0]全设置为0

顺序:先商品i后背包j(先背包后商品也行)

*/

int main()
{
    int n,m;
    cin >> n >> m;
    
    for(int i = 0; i < n; i++)
    {
        int tmpw,tmpval;
        cin >> tmpw >> tmpval;
        weight[i] = tmpw;
        val[i] = tmpval;
    }
    
    //初始化
    for(int j = 0; j <= m ; j++)
        if(j >= weight[0])f[0][j] = val[0];
        else f[0][j] = 0;
        
    for(int i = 0; i <= n; i++)f[i][0] = 0;
    
    for(int i = 1; i <= n; i++)
        for(int j = 1; j <= m; j++)
            if(j < weight[i])f[i][j] = f[i-1][j];  //装不下就只有不装一条路
            else f[i][j] = max(f[i-1][j],f[i-1][j-weight[i]]+val[i]);  //装的下就有装和不装两种选择

    //  for(int i = 0; i <= n; i++)
    // {
    //     for(int j = 0; j <= m; j++)
    //     {
    //         cout << f[i][j] << ' ';
    //     }
    //     puts("");
    // }
    
    cout << f[n][m];
    
    return 0;
}

1.3 01背包一维数组(滑动数组)

滑动数组实质就是利用01背包二维数组版本中每个f[i][j]只会用到上一层左侧 f[i-1][j-xx] 的数据,所以把上一层左侧数据保存在当前行的左侧,并且每次在当前行内遍历时候从右往左边遍历。AC代码:

#include<iostream>
using namespace std;

const int N = 1e3 + 10;

int weight[N];
int val[N];
int f[N];  // (j)表示容积j的背包能够装的最大价值

/*
转换方程:
if(j < weight[i])f[j] = f[j];  //装不下就只有不装一条路
else f[j] = max(f[j],f[j-weight[i]]+val[i]);  //装的下就有装和不装两种选择
              不装i       装i

第0行--f[0] = 0;

顺序:先商品i++后背包j--(j左边的作为上一层的记录),

*/

int main()
{
    int n,m;
    cin >> n >> m;
    
    for(int i = 0; i < n; i++)
    {
        int tmpw,tmpval;
        cin >> tmpw >> tmpval;
        weight[i] = tmpw;
        val[i] = tmpval;
    }
    
    //初始化
    for(int i = 0; i <= n; i++)
        for(int j = m; j >= 0; j--)
            if(j < weight[i])f[j] = f[j];  //装不下就只有不装一条路
            else f[j] = max(f[j],f[j-weight[i]]+val[i]);  //装的下就有装和不装两种选择


    
    cout << f[m];
    
    return 0;
}

2 01背包应用题1——416. 分割等和子集

416. 分割等和子集

一开始看到题目,想用贪心——排序+双指针 每次都把当前相对小的放进小的sum中,写完之后发现过不了:[1,1,2,2]这样的样例。错误代码:

class Solution {
public:
    /*
    左边的sum 小于 右边的sum l++,左边的sum+=
    左边的sum 大于 同理
    如果等于左边前进 
    1,2,3,4, 5,6,7,8,9, 10
    */
    bool canPartition(vector<int>& nums) 
    {
        // 解法1:排序+双指针
        if(nums.size() == 1)return 0;

        sort(nums.begin(),nums.end());

        int l = 0;
        int r = nums.size() - 1;
        int leftsum = 0;
        int rightsum = 0;
        while(l <= r)
        {
            if(leftsum <= rightsum)
            {
                leftsum += nums[l++];
            }
            else
            {
                rightsum += nums[r--];
            }
        }
        cout << l << "  " << r << endl;
        cout << leftsum << "  " << rightsum;
        if(leftsum == rightsum)return 1;
        else return 0;
    }
};

要明确本题中我们要使用的是01背包,因为元素我们只能用一次。

回归主题:首先,本题要求集合里能否出现总和为 sum / 2 的子集。

那么来一一对应一下本题,看看背包问题如何来解决。

只有确定了如下四点,才能把01背包问题套到本题上来。

  • 背包的体积为sum / 2
  • 背包要放入的商品(集合里的元素)重量为 元素的数值,价值也为元素的数值
  • 背包如果正好装满,说明找到了总和为 sum / 2 的子集。
  • 背包中每一个元素是不可重复放入。

具体分析过程见注释, AC代码:

class Solution {
public:
    // 找到一个背包 能够装nums.total(所有物体重量总和)/2的东西
    int dp[10005];    // 容积为i的背包 根据现有的物体重量情况最多能装的物体的重量
    /*
    转换成01背包问题:
    假设有一个nums.total/2的背包
    有若干个物体,每个物体的重量就是nums[i] 
    本题可以舍弃价值这个概念
    就是问一个nums.total/2的背包最多能够装的物体的重量是多少 能不能达到nums.total/2

    if(j < nums[i])dp[j] = dp[j];
    else dp[j] = max(dp[j] , dp[j - nums[i]]+nums[i]);

    dp[0] = 0;
    其他默认是0
    
    for物体i++ for容积j--

    模拟——

    */

    bool canPartition(vector<int>& nums) 
    {
        dp[0] = 0;

        int total = 0;
        for(auto i : nums)total += i;
        
        if(total % 2 == 0)total /= 2;
        else return 0;

        for(int i = 0; i < nums.size();i++)
        {
            for(int j = total ; j >= 0; j--)
            {
                if(j < nums[i])dp[j] = dp[j];
                else dp[j] = max(dp[j] , dp[j - nums[i]]+nums[i]);
            }
        }

        if(dp[total] == total)return 1;
        else return 0;

    }
};

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

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

相关文章

JavaSE 集合框架及背后的数据结构

目录 1 介绍2 学习的意义2.1 Java 集合框架的优点及作用2.2 笔试及面试题 3 接口 interfaces3.1 基本关系说明3.2 Collection 常用方法说明3.3 Collection 示例3.4 Map 常用方法说明3.5 Map 示例 4 实现 classes5 Java数据结构知识体系5.1 目标5.2 知识点 1 介绍 集合&#xf…

【运维】hadoop集群安装(一)多节点安装

文章目录 一.Purpose二. Prerequisites三. Installation1. 节点规划2. Configuring Hadoop in Non-Secure Mode3. 准备工作4. 配置core-site.xmlhdfs-site.xmlyarn-site.xmlmapred-site.xmlworkers 4. 分发配置、创建文件夹5. 格式化6. 操作进程6.1. hdfs启动停止 6.2. yarn启动…

java 高级面试题整理(薄弱技术-2023)

session 和cookie的区别和联系 session1.什么是session Session是另一种记录客户状态的机制&#xff0c;不同的是Cookie保存在客户端浏览器中&#xff0c;而Session保存在服务器上。客户端浏览器访问服务器的时候&#xff0c;服务器把客户端信息以某种形式记录在服务器上。这就…

Docker容器:docker consul的注册与发现及consul-template

Docker容器&#xff1a;docker consul的注册与发现及consul-template守护进程 一.docker consul的注册与发现介绍 1.什么是服务注册与发现 &#xff08;1&#xff09;服务注册与发现是微服务架构中不可或缺的重要组件。 &#xff08;2&#xff09;为解决服务都是单节点的&a…

基于动物迁徙算法优化的BP神经网络(预测应用) - 附代码

基于动物迁徙算法优化的BP神经网络&#xff08;预测应用&#xff09; - 附代码 文章目录 基于动物迁徙算法优化的BP神经网络&#xff08;预测应用&#xff09; - 附代码1.数据介绍2.动物迁徙优化BP神经网络2.1 BP神经网络参数设置2.2 动物迁徙算法应用 4.测试结果&#xff1a;5…

IT论坛测试

目录 一、项目介绍 项目名称 项目简介 相关技术 项目展示 二 、测试用例设计和功能测试 测试用例设计 注册页面 登陆页面 首页面 发布帖子页面 修改个人信息页面 功能测试 注册页面 登录页面 首页面 发布帖子页面 修改个人信息页面 三、接口测试 1.Junit单…

HAproxy+keepalived高可用配置搭建

目录 一、概述 &#xff08;一&#xff09;简介 &#xff08;二&#xff09;核心功能 &#xff08;三&#xff09;关键特性 &#xff08;四&#xff09;应用场景 二、安装 1&#xff09;拓补图 2&#xff09;配置 &#xff08;一&#xff09;内核配置 &#xff08;二…

【golang】派生数据类型---指针 标识符、关键字等

1、指针 对比C/C中的指针&#xff0c;go语言中的指针显得极为简洁&#xff0c;只是简单的获取某个空间的地址 或者 根据指针变量中的内容 获取对应存储空间的内容等操作。 具体示例如下&#xff1a; go中使用指针需要注意的点&#xff1a; 可以通过指针改变它所指向的内存空…

5 STM32标准库函数 之 外部中断/事件控制器(EXTI)所有函数的介绍及使用

5 STM32标准库函数 之 外部中断/事件控制器&#xff08;EXTI&#xff09;所有函数的介绍及使用 1. 图片有格式2 文字无格式五 库函数之外部中断/事件控制器&#xff08;EXTI&#xff09;所有函数的介绍及使用前言一、图片预览&#xff0c;无格式&#xff08;CSDN&#xff09;二…

C++ 工具

C参考手册 Learncppcplusplustutorialspoint/cplusplusAwesomeC 写在最后&#xff1a;若本文对您有帮助&#xff0c;请点个赞啦 ٩(๑•̀ω•́๑)۶

RS485保护电路

今天给大家分享485接口的EMC设计&#xff0c;希望对电路设计&#xff0c;及相关软件开发的人员有帮助。 一、原理图 1. RS485接口6KV防雷电路设计方案 &#xff08;RS485接口防雷电路&#xff09; 接口电路设计概述&#xff1a; RS485用于设备与计算机或其它设备之间通讯&…

c++11 标准模板(STL)(std::basic_ostringstream)(一)

定义于头文件 <sstream> template< class CharT, class Traits std::char_traits<CharT> > class basic_ostringstream;(C11 前)template< class CharT, class Traits std::char_traits<CharT>, class Allocator std::allo…

week6刷题

题解: 使用二分查找 class Solution { public:int minArray(vector<int>& numbers) {int low 0;int high numbers.size() - 1;while (low < high) {int pivot low (high - low) / 2;if (numbers[pivot] < numbers[high]) {high pivot;}else if (numbers…

学习JAVA打卡第四十四天

Scanner类 ⑴Scanner对象 scanner对象可以解析字符序列中的单词。 例如&#xff1a;对于string对象NBA 为了解析出NBA的字符序列中的单词&#xff0c;可以如下构造一个scanner对象。 将正则表达式作为分隔标记&#xff0c;即让scanner对象在解析操作时把与正则表达式匹配的字…

<C++> STL_stack和queue

1.stack的介绍和使用 1.1 stack的介绍 stack是一种容器适配器&#xff0c;专门用在具有后进先出操作的上下文环境中&#xff0c;其删除只能从容器的一端进行元素的插入与提取操作。 stack是作为容器适配器被实现的&#xff0c;容器适配器即是对特定类封装作为其底层的容器&am…

数据生成 | MATLAB实现GAN生成对抗网络结合SVM支持向量机的数据生成

数据生成 | MATLAB实现GAN生成对抗网络结合SVM支持向量机的数据生成 目录 数据生成 | MATLAB实现GAN生成对抗网络结合SVM支持向量机的数据生成生成效果基本描述程序设计参考资料 生成效果 基本描述 数据生成 | MATLAB实现GAN生成对抗网络结合SVM支持向量机的数据生成。 生成对抗…

抖店的出单主要靠什么?给你们分析下抖店的运营逻辑,建议看完!

我是王路飞。 做抖店只有一个目的&#xff0c;就是出单、赚钱。 那么从开通抖店开始&#xff0c;到最后店铺出单&#xff0c;这中间的一些运营流程就是你出单的关键了。 那么抖店的出单主要靠什么呢&#xff1f; 我的观点是&#xff1a;产品流量。给你们分析下抖店的运营逻…

【JavaEE】Spring事务-@Transactional参数介绍-事务的隔离级别以及传播机制

【JavaEE】Spring事务&#xff08;2&#xff09; 文章目录 【JavaEE】Spring事务&#xff08;2&#xff09;1. Transactional 参数介绍1.1 value 和 transactionManager1.2 timeout1.3 readOnly1.4 后面四个1.5 isolation 与 propagation 2. Spring 事务隔离级别 - isolation2.…

登录校验-Filter-登录校验过滤器

目录 思路 登录校验Filter-流程 步骤 流程图 登录校验Filter-代码 过滤器类 工具类 测试登录 登录接口功能请求 其他接口功能请求 前后端联调 思路 前端访问登录接口&#xff0c;登陆成功后&#xff0c;服务端会生成一个JWT令牌&#xff0c;并返回给前端&#xff0…

【C++】C++ 引用详解 ⑦ ( 指针的引用 )

文章目录 一、二级指针可实现的效果二、指针的引用1、指针的引用 等同于 二级指针 ( 重点概念 )2、引用本质 - 函数间接赋值简化版本3、代码示例 - 指针的引用 一、二级指针可实现的效果 指针的引用 效果 等同于 二级指针 , 因此这里先介绍 二级指针 ; 使用 二级指针 作为参数 …