DP:二维费用背包问题

news2024/11/14 19:03:50

文章目录

  • 🎵二维费用背包问题
    • 🎶引言
    • 🎶问题定义
    • 🎶动态规划思想
    • 🎶状态定义和状态转移方程
    • 🎶初始条件和边界情况
  • 🎵例题
    • 🎶1.一和零
    • 🎶2.盈利计划
  • 🎵总结

在这里插入图片描述

在这里插入图片描述

🎵二维费用背包问题

🎶引言

背包问题是算法中的经典问题之一,其变种繁多。本文将介绍二维费用背包问题及其解决方案。

🎶问题定义

二维费用背包问题可以描述为:给定 (N) 个物品,每个物品有两个费用和一个价值,在两种费用的限制下,如何选择物品使得总价值最大。

🎶动态规划思想

动态规划是解决背包问题的常用方法。通过定义合适的状态和状态转移方程,我们可以有效地解决二维费用背包问题。

🎶状态定义和状态转移方程

我们定义 dp[i][j][k] 表示前 i 个物品在费用1不超过 j 和费用2不超过 k 的情况下的最大价值。状态转移方程为:

d p [ i ] [ j ] [ k ] = max ⁡ ( d p [ i − 1 ] [ j ] [ k ] , d p [ i − 1 ] [ j − c o s t 1 [ i ] ] [ k − c o s t 2 [ i ] ] + v a l u e [ i ] ) dp[i][j][k] = \max(dp[i-1][j][k], dp[i-1][j-cost1[i]][k-cost2[i]] + value[i]) dp[i][j][k]=max(dp[i1][j][k],dp[i1][jcost1[i]][kcost2[i]]+value[i])

🎶初始条件和边界情况

初始条件为 dp[0][j][k] = 0,表示没有物品时的最大价值为 0。边界条件需要根据实际问题进行处理。

🎵例题

🎶1.一和零

题目:

在这里插入图片描述

样例输出和输入:

在这里插入图片描述

算法原理:
这道题就是让我们找子集的长度,这个子集满足:当中的0不大于m个,当中的1不大于n个,最后返回最大的子集的长度,所以我们首先想到的是二维费用背包问题,因为有两个限制,这里的背包的限制就是0和1的个数的限制,这里的物品其实就是每个字符串。
状态表示: d p [ i ] [ j ] [ k ] dp[i][j][k] dp[i][j][k]表示从前 i i i个物品中选择的所有组合中,满足0的个数不大于m,1的个数不大于n个的所有组合中子集长度最大的那个的长度。
状态转移方程:
在这里插入图片描述
这里的a和b代表的是当前i位置字符串中0和1分别的个数,所以我们在进行填表的时候应该遍历一下字符串,将当中的0和1分别记录一下,状态转移方程:

d p [ i ] [ j ] [ k ] = m a x ( d p [ i − 1 ] [ j ] [ k ] , d p [ i − 1 ] [ j − a ] [ k − b ] ) dp[i][j][k]=max(dp[i-1][j][k],dp[i-1][j-a][k-b]) dp[i][j][k]=max(dp[i1][j][k],dp[i1][ja][kb])

初始化:

代码:
未优化的代码:

class Solution {
public:
    int findMaxForm(vector<string>& strs, int m, int n)
    {
        int sz = strs.size();
        vector<vector<vector<int>>> dp(sz + 1, vector<vector<int>>(m + 1, vector<int>(n + 1)));
        for (int i = 1;i <= sz;i++)
        {
            //统计一下字符串中0和1的个数
            int a = 0, b = 0;
            for (auto e : strs[i - 1])
            {
                if (e == '1')b++;
                else a++;
            }
            for (int j = 0;j <= m;j++)
            {
                for (int k = 0;k <= n;k++)
                {
                    dp[i][j][k] = dp[i - 1][j][k];
                    if (j >= a && k >= b)dp[i][j][k] = max(dp[i - 1][j][k], dp[i - 1][j - a][k - b] + 1);
                }
            }
        }
        return dp[sz][m][n];
    }
};

滚动数组优化的代码:

class Solution {
public:
	int findMaxForm(vector<string>& strs, int m, int n)
	{
		int sz = strs.size();
		vector<vector<int>> dp(m + 1, vector<int>(n + 1));
		for (int i = 1;i <= sz;i++)
		{
			//统计一下字符串中0和1的个数
			int a = 0, b = 0;
			for (auto e : strs[i - 1])
			{
				if (e == '1')b++;
				else a++;
			}
			for (int j = m;j >=a;j--)
				for (int k = n;k >=b;k--)
				dp[j][k] = max(dp[j][k], dp[j - a][k - b] + 1);
		}
		return dp[m][n];
	}
};

运行结果:
在这里插入图片描述

🎶2.盈利计划

题目:

在这里插入图片描述

样例输出和输入:

在这里插入图片描述

算法原理:
这道题每个group对应一个profit,下标是对应的。
在这里插入图片描述
根据上面的图片加上题目要求,我们可以得知,我们每次选择的利润必须大于给定的 m i n P r o f i t minProfit minProfit然后每次需要的人口不能超过 n n n,最后求出满足这个条件的所有组合有多少种。
状态表示: d p [ i ] [ j ] [ k ] dp[i][j][k] dp[i][j][k]表示从前i个工作计划中选择,人数不超过i的,但是盈利大于k的所有组合数的总和。
状态转移方程:
第一种状态:不选择i位置, d p [ i − 1 ] [ j ] [ k ] dp[i-1][j][k] dp[i1][j][k]

第二种状态:选择i位置,首先考虑二维 d p [ i − 1 ] [ j − g r o u p [ i ] ] dp[i-1][j-group[i]] dp[i1][jgroup[i]]这里我们考虑一下 j − g r o u p [ i ] ≤ 0 j-group[i]\leq0 jgroup[i]0是否成立将group[i]移到右边去可以得到: j ≤ g r o u p [ i ] j\leq group[i] jgroup[i]这个是什么意思呢?表示i工作需要的人口是大于总人口j的,所以这肯定是不可能的,所以这里中只能是 j − g r o u p [ i ] ≥ 0 j-group[i]\geq0 jgroup[i]0,我们再来考虑三维的: d p [ i − 1 ] [ j − g r o u p [ i ] ] [ k − p r o f i t [ i ] ] dp[i-1][j-group[i]][k-profit[i]] dp[i1][jgroup[i]][kprofit[i]]我们来考虑 k − p r o f i t [ i ] ≤ 0 k-profit[i]\leq0 kprofit[i]0是否成立,首先我们还是继续移一下项: k ≤ p r o f i t [ i ] k \leq profit[i] kprofit[i]这里k表示总的利润,profit表示当前工作产出的利润,所以这里的意思就表示无论前面总利润是多少,这里都都能满足当前的利润,所以我们只需要选择0即可,所以第二种状态:

d p [ i − 1 ] [ j − g r o u p [ i ] ] [ m a x ( 0 , k − p r o f i t [ i ] ) ] dp[i-1][j-group[i]][max(0,k-profit[i])] dp[i1][jgroup[i]][max(0,kprofit[i])]

最后这两种状态的总和就是当前状态的所有组合的总和:

d p [ i ] [ j ] [ k ] = d p [ i − 1 ] [ j ] [ k ] + d p [ i − 1 ] [ j − g r o u p [ i ] ] [ m a x ( 0 , k − p r o f i t [ i ] ) ] dp[i][j][k]=dp[i-1][j][k]+dp[i-1][j-group[i]][max(0,k-profit[i])] dp[i][j][k]=dp[i1][j][k]+dp[i1][jgroup[i]][max(0,kprofit[i])]

代码:
未优化的代码:

class Solution {
public:
    int profitableSchemes(int n, int minProfit, vector<int>& group, vector<int>& profit) 
    {
        int len = group.size();
        int MOD = 1e9 + 7;
        vector<vector<vector<int>>> dp(len + 1, vector<vector<int>>(n + 1, vector<int>(minProfit + 1)));
        for (int j = 0;j <= n;j++)
        {
            dp[0][j][0] = 1;
        }
        for (int i = 1;i <= len;i++)
        {
            for (int j = 0;j <= n;j++)
            {
                for (int k = 0;k <= minProfit;k++)
                {
                    dp[i][j][k] = dp[i - 1][j][k];
                    if (j >= group[i-1])
                        dp[i][j][k] += dp[i - 1][j - group[i - 1]][max(0, k - profit[i - 1])];
                    dp[i][j][k] %= MOD;
                }
            }
        }
        return dp[len][n][minProfit];
    }
};

优化过后的代码:

int profitableSchemes(int n, int minProfit, vector<int>& group, vector<int>& profit) 
{
	int len = group.size();
	int MOD = 1e9 + 7;
	vector<vector<int>> dp(n + 1, vector<int>(minProfit + 1));
	for (int j = 0;j <= n;j++)dp[j][0] = 1;
	for (int i = 1;i <= len;i++)
		for (int j = n;j >= group[i - 1];j--)
			for (int k = 0;k <= minProfit;k++)
			{
				dp[j][k] += dp[j - group[i - 1]][max(0, k - profit[i - 1])];
				dp[j][k] %= MOD;
			}
	return dp[n][minProfit];
}

运行结果:
在这里插入图片描述

🎵总结

通过本文的学习,我们了解了二维费用背包问题的基本概念和解决方法。与传统的单一费用背包问题不同,二维费用背包问题在解决时需要同时考虑两种费用的限制,这使得问题更具挑战性,但也更贴近实际应用场景。我们通过动态规划的方法,逐步构建状态转移方程,并利用二维数组来存储中间结果,从而实现了对问题的高效求解。

在实际应用中,掌握二维费用背包问题的解决思路,可以帮助我们在资源分配、投资组合等多方面优化决策。希望通过本次的学习,大家不仅能够掌握相关的算法技巧,还能够举一反三,灵活应用于更多复杂的优化问题中。

今后,我们将继续探讨更为复杂的动态规划问题,进一步提升算法设计和问题求解能力。谢谢大家的阅读,希望本文对你有所帮助。

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

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

相关文章

Explore Synapse

rm -r dp-203 -f git clone https://github.com/MicrosoftLearning/dp-203-azure-data-engineer dp-203 cd dp-203/Allfiles/labs/01 ./setup.ps1 -- This is auto-generated code SELECTTOP 100 * FROMOPENROWSET(BULK https://datalakexxxxxxx.dfs.core.windows.net/fil…

单/多线程--协程--异步爬虫

免责声明:本文仅做技术交流与学习... 目录 了解进程和线程 单个线程(主线程)在执行 多线程 线程池 协程(爬虫多用) 假异步:(同步) 真异步: 爬虫代码模版 异步-爬虫 同步效果--19秒 异步效果--7秒 了解进程和线程 ​ # --------------------> # ------> # …

Opencv的基本操作(一)图像的读取显示存储及几何图形的绘制

文件的读取、显示、存取 cv2.imread(imagepath,IMREAD.xxx) 读取图像cv2.imshow(窗口名称,mat图片) 显示图像cv2.imwrite(保存的位置,img) 保存图像 # 1. 读取图像 原始图片路径&#xff0c;图片读取模式 cv2.imread(imagepath,IMREAD.xxx)cv2.IMREAD_COLOR 彩色模式读取 cv2…

PostgreSQL 如何优化存储过程的执行效率?

文章目录 一、查询优化1. 正确使用索引2. 避免不必要的全表扫描3. 使用合适的连接方式4. 优化子查询 二、参数传递1. 避免传递大对象2. 参数类型匹配 三、减少数据量处理1. 限制返回结果集2. 提前筛选数据 四、优化逻辑结构1. 分解复杂的存储过程2. 避免过度使用游标 五、事务处…

贵州建筑三类人员安全员2024年考试最新题库练习题

一、单选题 1.建设工程安全管理的方针是&#xff08;&#xff09;。 A.安全第一&#xff0c;预防为主&#xff0c;综合治理 B.质量第一&#xff0c;兼顾安全 C.安全至上 D.安全责任重于泰山 答案&#xff1a;A 2.安全生产管理的根本目的是&#xff08;&#xff09;。 A.…

zerotier-one自建根服务器方法五

一、简介 前面几篇文章已经写完了自己建立服务器的方法&#xff0c;今天写一下我在使用过程中遇到的问题和解决方法。 二、准备工作 准备一个有公网IP的云主机。 要稳定性、安全性、不差钱的可以使用阿里、腾讯等大厂的云服务器。 本人穷屌丝一枚&#xff0c;所以我用的是免…

数据结构1:C++实现边长数组

数组作为线性表的一种&#xff0c;具有内存连续这一特点&#xff0c;可以通过下标访问元素&#xff0c;并且下标访问的时间复杂的是O(1)&#xff0c;在数组的末尾插入和删除元素的时间复杂度同样是O(1)&#xff0c;我们使用C实现一个简单的边长数组。 数据结构定义 class Arr…

CentOS 7.9 停止维护(2024-6-30)后可用在线yum源 —— 筑梦之路

众所周知&#xff0c;centos 7 在2024年6月30日&#xff0c;生命周期结束&#xff0c;官方不再进行支持维护&#xff0c;而很多环境一时之间无法完全更新替换操作系统&#xff0c;因此对于yum源还是需要的&#xff0c;特别是对于互联网环境来说&#xff0c;在线yum源使用方便很…

第6章 选课学习:需求分析,添加选课,支付,支付通知,在线学习

1 模块需求分析 1.1 模块介绍 本模块实现了学生选课、下单支付、学习的整体流程。 网站的课程有免费和收费两种&#xff0c;对于免费课程学生选课后可直接学习&#xff0c;对于收费课程学生需要下单且支付成功方可选课、学习。 选课&#xff1a;是将课程加入我的课程表的过…

关于HTTP的攻击实验

实验原理&#xff1a;1. 根据ARP中间人攻击&#xff0c;获取 用户和服务器之间的数据2. 将获取到的数据 通过一定的技术来复原&#xff0c;进而获取用户的信息或者 相关权限实验拓扑图 将 kali 的网卡改为桥接模式&#xff0c;查看Kali和本机的ip 启动ettercap&#xff0c;…

CANopen协议开发梳理总结笔记教程

0、提醒 CANOpen使用时&#xff0c;需要清楚什么是大端和小端&#xff0c;这对于CANOpen数据发送及解析时&#xff0c;有很大的帮助。且学习开发CANOpen时&#xff0c;需要具备一定的CAN基础。 1、CANOpen协议介绍 ①、什么是CANOpen协议 CANOpen协议是一种架构在控制局域网络…

FreeRTOS——队列集

一、队列集 一个队列只允许任务间传递的消息为 同一种数据类型 &#xff0c;如果需要在任务间 传递不同数据类型的消息 时&#xff0c;那么就可以使用队列集 作用&#xff1a;用于对多个队列或信号量进行“监听”&#xff08;接收或获取&#xff09;&#xff0c;其中 不管哪一…

Java线上接口耗时分析神器 Arthas

介绍 程序员的日常&#xff0c;总是离不开“调优”和“排查”。尤其当线上环境出现问题&#xff0c;性能瓶颈把人逼疯。这时候&#xff0c;你就需要一款像 Arthas 这样的神器来救场。 什么是 Arthas&#xff1f; 简单来说&#xff0c;Arthas 是阿里巴巴开源的 Java 诊断工具…

SwinUnet详解

文章目录 摘要一. 编码端模块1. PatchEmbed2. SwinTransformerBlock2.1. Window_partition2.2. WindowAttention2.3. Window_reverse2.4. MLP 3. PatchMerging 二. 解码端模块三. 完整流程图 摘要 swinunet基本结构&#xff1a; swinunet采用编码器-解码器结构&#xff1a; 编…

python读取csv出错怎么解决

Python用pandas的read_csv函数读取csv文件。 首先&#xff0c;导入pandas包后&#xff0c;直接用read_csv函数读取报错OSError&#xff0c;如下&#xff1a; 解决方案是加上参数&#xff1a;enginepython。 运行之后没有报错&#xff0c;正在我欣喜之余&#xff0c;输出一下d…

vulnhub靶场之Jarbas

1 信息收集 1.1 主机发现 arp-scan -l 发现主机IP地址为&#xff1a;192.168.1.16 1.2 端口发现 nmap -sS -sV -A -T5 -p- 192.168.1.16 存在端口22&#xff0c;80&#xff0c;3306&#xff0c;8080 1.3 目录扫描 dirsearch -u 192.168.1.16 2 端口访问 2.1 80端口 2.2…

软件设计之Java入门视频(12)

软件设计之Java入门视频(12) 视频教程来自B站尚硅谷&#xff1a; 尚硅谷Java入门视频教程&#xff0c;宋红康java基础视频 相关文件资料&#xff08;百度网盘&#xff09; 提取密码&#xff1a;8op3 idea 下载可以关注 软件管家 公众号 学习内容&#xff1a; 该视频共分为1-7…

【OnlyOffice】桌面应用编辑器,插件开发大赛,等你来挑战

OnlyOffice&#xff0c;桌面应用编辑器&#xff0c;最近版本已从8.0升级到了8.1 从PDF、Word、Excel、PPT等全面进行了升级。随着AI应用持续的火热&#xff0c;OnlyOffice也在不断推出AI相关插件。 因此&#xff0c;在此给大家推荐一下OnlyOffice本次的插件开发大赛。 详细信息…

[激光原理与应用-98]:南京科耐激光-激光焊接-焊中检测-智能制程监测系统IPM介绍 - 2 - 什么是激光器焊接? 常见的激光焊接技术详解

目录 一、什么是激光焊接 1.1 概述 1.2 激光焊接的优点 二、激光焊接的应用 2.1 哪些场合必须使用激光焊接 1. 汽车制造业 2. 航空航天领域 3. 电子行业&#xff1a;消费类电子3C 4. 医疗器械制造 5. 新能源锂电池行业 6. 其他领域 三、激光焊接的分类 3.1 按焊接…

指针回顾.

指针的主要作用&#xff1a;提供一种间接访问数据的方法 1.地址:区分不同内存空间的编号 2.指针:指针就是地址,地址就是指针 3.指针变量:存放指针的变量称为指针变量,简称为指针 1.指针的定义 int *p NULL; int *q NULL; char *p NULL; double *p NUL…