DP:二维费用背包问题

文章目录

  • 🎵二维费用背包问题
    • 🎶引言
    • 🎶问题定义
    • 🎶动态规划思想
    • 🎶状态定义和状态转移方程
    • 🎶初始条件和边界情况
  • 🎵例题
    • 🎶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.mfbz.cn/a/778344.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

身体(body)的觉醒:如果你贪婪,给你整个宇宙都不够

佛&#xff0c;是一个梵文的汉语音译词&#xff0c;指觉醒者。 何谓觉醒&#xff1f;什么的觉醒&#xff1f;其实很简单&#xff0c;就是身体的觉醒。 佛的另一个名字&#xff0c;叫菩提&#xff0c;佛就是菩提&#xff0c;菩提老祖&#xff0c;就是佛祖。 一、body&#xff…

跟《经济学人》学英文:2024年07月06日这期:Finishing schools for the age of TikTok

Finishing schools for the age of TikTok Unsure how to be polite at work? Ask a digital etiquette guru 不确定如何在工作中保持礼貌&#xff1f;请教一位数字礼仪大师 “Finishing schools” 是指专门为年轻女性提供礼仪、社交技巧、文化修养等教育的学校&#xff0c;…

WordPress网站添加插件和主题时潜在危险分析

WordPress 最初只是一个简单的博客软件&#xff0c;现在据估计为全球前 1000 万个网站中的 30% 提供支持。WordPress受欢迎的因素之一是可以轻松创建插件和主题来扩展它并提供比默认设置更多的功能。 目前&#xff0c;WordPress 网站列出了 56,000 多个插件以及数千个主题。插件…

【刷题汇总--大数加法、 链表相加(二)、大数乘法】

C日常刷题积累 今日刷题汇总 - day0061、大数加法1.1、题目1.2、思路1.3、程序实现 2、 链表相加(二)2.1、题目2.2、思路2.3、程序实现 3、大数乘法3.1、题目3.2、思路3.3、程序实现 4、题目链接 今日刷题汇总 - day006 1、大数加法 1.1、题目 1.2、思路 读完题,明白大数相加…

【C++干货基地】C++模板深度解析:进阶技巧与高级特性掌握(按需实例化、全特化与偏特化)文末送书

&#x1f3ac; 鸽芷咕&#xff1a;个人主页 &#x1f525; 个人专栏: 《C干货基地》《粉丝福利》 ⛺️生活的理想&#xff0c;就是为了理想的生活! 引入 哈喽各位铁汁们好啊&#xff0c;我是博主鸽芷咕《C干货基地》是由我的襄阳家乡零食基地有感而发&#xff0c;不知道各位的…

带你解刨自动化测试框架详细总结

自动化测试框架概念 自动化测试框架是一个集成体系&#xff0c;这个体系中包含测试功能的函数库、测试数据源、测试对象以及可重用的模块。 框架&#xff08;framework&#xff09;是一个框子——指其约束性&#xff0c;也是一个架子——指其支撑性。是一个基本概念上的结构&…

go zero入门

一、goctl安装 goctl 是 go-zero 的内置脚手架&#xff0c;可以一键生成代码、文档、部署 k8s yaml、dockerfile 等。 # Go 1.16 及以后版本 go install github.com/zeromicro/go-zero/tools/goctllatest检查是否安装成功 $ goctl -v goctl version 1.6.6 darwin/amd64vscod…

String类对象比较:==和equals的具体细节

public class test {public static void main(String[] args) {String name1 "zzz";String name2 "zzz";String name3 new String("zzz");// hashCode() 方法&#xff1a;基于字符串的内容计算哈希值&#xff0c;因此内容相同的字符串对象其 …

macOS查看系统日志的方法

1、command空格键打开搜索框&#xff0c;输入‘控制台’并打开 2、选择日志报告&#xff0c;根据日期打开自己需要的文件就可以

ESP32——物联网小项目汇总

商品级ESP32智能手表 [文章链接] 用ESP32&#xff0c;做了个siri&#xff1f;&#xff01;开源了&#xff01; [文章链接]

win11中配制了系统的环境变量mvn/java,但是mvn/java就是提示不存在的解决方法。

1、已经配制了环境变量&#xff0c;但是提示mvn不存在 2、然后我们在开始程序中查看到cmd&#xff0c;然后以管理员运行&#xff1a; 这样的话&#xff0c;是可以mvn这个命令的&#xff0c;而且只有这种方式是可以的&#xff0c;其它的方式&#xff0c;就算设置了以管理员身份运…

Canary,三种优雅姿势绕过

Canary&#xff08;金丝雀&#xff09;&#xff0c;栈溢出保护 canary保护是防止栈溢出的一种措施&#xff0c;其在调用函数时&#xff0c;在栈帧的上方放入一个随机值 &#xff0c;绕过canary时首先需要泄漏这个随机值&#xff0c;然后再钩爪ROP链时将其作为垃圾数据写入&…

编程上下文Context及其实现原理

编程上下文Context及其实现原理 author:shengfq date:2024-07-06 title:编程上下文Context及其实现原理 category:编程思想1.编程中的上下文Context是指什么? 在编程和软件工程领域&#xff0c;“上下文”&#xff08;Context&#xff09;是一个多义词&#xff0c;其含义可以…

开始尝试从0写一个项目--后端(二)

实现学生管理 新增学生 接口设计 请求路径&#xff1a;/admin/student 请求方法&#xff1a;POST 请求参数&#xff1a;请求头&#xff1a;Headers&#xff1a;"Content-Type": "application/json" 请求体&#xff1a;Body&#xff1a; id 学生id …

VideoAgent——使用大规模语言模型作为代理来理解长视频

概述 论文地址&#xff1a;https://arxiv.org/pdf/2403.10517 本研究引入了一个新颖的基于代理的系统&#xff0c;名为 VideoAgent。该系统以大规模语言模型为核心&#xff0c;负责识别关键信息以回答问题和编辑视频。VideoAgent 在具有挑战性的 EgoSchema 和 NExT-QA 基准上进…

MySQL架构和工作流程

引言&#xff1a;MySQL执行一条sql语句期间发生了什么&#xff1f; 想要搞清楚这个问题&#xff0c;我们必须了解MySQL的体系结构和工作流程 一、MySQL体系结构 MySQL由以下几个部分组成 一、server层 1.MySQL Connnectors连接器&#xff0c;MySQL的连接池组件&#xff0c;…

【vue组件库搭建05】vitePress中使用vue/antd/demo预览组件

一、vitepress使用vue及antd组件 1.安装antd之后在docs\.vitepress\theme\index.ts引入文件 // https://vitepress.dev/guide/custom-theme import { h } from vue import type { Theme } from vitepress import DefaultTheme from vitepress/theme import ./style.css impor…

React 19 竞态问题解决

竞态问题/竞态条件 指的是&#xff0c;当我们在交互过程中&#xff0c;由于各种原因导致同一个接口短时间之内连续发送请求&#xff0c;后发送的请求有可能先得到请求结果&#xff0c;从而导致数据渲染出现预期之外的错误。 因为防止重复执行可以有效的解决竞态问题&#xff0…

试用笔记之-汇通Exe可执行文件之pe分析

首先下载汇通Exe可执行文件之pe分析 http://www.htsoft.com.cn/download/pedump.rar

苹果笔记本能玩网页游戏吗 苹果电脑玩steam游戏怎么样 苹果手机可以玩游戏吗 mac电脑安装windows

苹果笔记本有着优雅的机身、强大的性能&#xff0c;每次更新迭代都备受用户青睐。但是&#xff0c;当需要使用苹果笔记本进行游戏时&#xff0c;很多人会有疑问&#xff1a;苹果笔记本能玩网页游戏吗&#xff1f;苹果笔记本适合打游戏吗&#xff1f;本文将讨论这两个话题&#…