动态规划(基础)

news2025/1/16 19:05:11

目录

一、算法思想

二、解题步骤 

三、神奇的兔子序列 

(一)问题

(二)递归公式

(三)以求解F(6)为例 

(四)代码

四、01背包问题

(一)算法思想 

(二)举例  

1. 有3种物品

2. 背包问题网格

3. 初始化第一列

4. 吉他行

5. 音箱行 

6. 电脑行

7. 总结

(三)核心代码  

(四)完整代码  


一、算法思想

  • 动态规划也是一种分治思想,但与分治算法不同的是,分治算法是把原问题分解成若干子问题,自顶向下求解各子问题,合并子问题的解,从而得到原问题的解。动态规划也是把原问题分解为若干子问题然后自底向上,先求解最小的子问题,把结果存储在表格中,再求解大的子问题时,直接从表格中查询小的子问题的解,避免重复计数,从而提高算法效率。

二、解题步骤 

(1)分析最优解的结构特征

(2)建立最优值的递归式

(3)自底向上计算最优值,并记录

(4)构造最优解

三、神奇的兔子序列 

(一)问题

前两个月都是1对兔子,从第3个月开始,当月的兔子数等于前两个月的兔子数,求解第n个月的兔子数量?

(二)递归公式

 F(n)=\begin{cases}1 & n = 1,n=2\\F(n-1)+F(n-2) & n > 2\end{cases}

(三)以求解F(6)为例 

  • 利用动态规划求解的时候,记录结果,重复问题只需要求解依次即可

(四)代码

int Fib(int n)
{
	if (n < 1)
		return -1;
	else
	{
		vector<int>F(n+1);
		F[0]=0;
		F[1]=1;
		F[2]=1;
		for (int i = 3; i <= n; i++)
		{
			F[i] = F[i - 1] + F[i - 2];
		}
		return F[n];
	}
}

四、01背包问题

(一)算法思想 

  • 用数组dp[i][j]表示从下标为0到i-1的物品里任意选取,然后放进容量为j的背包中,背包所能装入的最大价值。

对于一个物品i,要么能放入背包,要么不能放入背包:

  • 第一种情况:物品i的重量w[i]大于背包容量j的时候,此时物品i不能放入背包中,那么需要从剩下0到i-1个物品中任意选取放入背包中,此时背包所能装入最大价值为dp[i][j]=dp[i-1][j]
  • 第二种情况:物品i的重量w[i]小于背包容量的时候,此时物品i能放入背包中,但是可以选择将物品i不放入背包或者放入背包。第二种情况又分以下两种:
  • 1.选择不将物品i放入背包的时候,那么从剩下的0到i-1个物品中任意选取价值大的物品放入背包中,此时背包所能装入最大价值为dp[i][j]=dp[i-1][j]
  • 2.选择将物品i放入背包的时候,需要先将物品i放入背包,那么背包剩余容量为j-w[i]
  • 此时从剩下的0到i-1个物品中任意选取价值大的物品放入背包中,背包所能装入最大价值为dp[i][j]=dp[i-1][j-w[i]]+v[i]
  • 那么物品最大价值为dp[i][j]=max\left \{ dp[i-1][j],dp[i-1][j-w[i]]+v[i] \right \}

所以背包所能装入最大价值为:

dp[i][j]=\begin{cases}dp[i-1][j] &,0<= j<w[i]\\max \left \{ dp[i-1][j] ,dp[i-1][j-w[i]]+v[i]\right \}&, j>=w[i]\end{cases}

(二)举例  

1. 有3种物品

物品价值重量
吉他1500美元1磅
音箱3000美元4磅
笔记本电脑2000美元3磅

2. 背包问题网格

 

3. 初始化第一列

  • 背包容量为0,意味着吉他,音箱,电脑都不能装入背包,所有此时背包最大价值均为0 

4. 吉他行

 1.  第2个单元格表示背包的容量为1磅。吉他的重量也是1磅,这意味着它能装入背包!

 

2. 之后背包容量为2磅,3磅,4磅也能装下

 3. 表明若一个背包容量为4磅,可在其中装入的商品的最大价值为1500美元

5. 音箱行 

 

 

  • 更新最大值:表明若一个背包容量为4磅,可在其中装入的商品的最大价值为3000美元  

6. 电脑行 

 

 

7. 总结

(三)核心代码  

template<typename T1,typename T2>
void Knapsack(int bagW,int n)
{
	vector<T1>value(n);//存放物品价值
	vector<T2>weight(n);//物品数量
	Init(n,value,weight);//初始化物品信息
	vector<vector<T1>>dp(n,vector<T1>(bagW+1,0));//背包最大价值

	for (int i = 0; i < n; i++)//初始化第一列
	{
		dp[i][0] = 0;
	}
	for (int j = 1; j <= bagW; j++)//初始化第一行
	{
		if (j < weight[0])//背包容量小于物品重量
			dp[0][j] = 0;
		else
			dp[0][j] = value[0];
	}

	for (int i = 1; i < n; i++)//i是物品序号
	{
		for (int j = 1; j <=bagW; j++)//j表示背包容量
		{
			if (j < weight[i])//容量为j的背包装不了物品i
				dp[i][j] = dp[i - 1][j];//此时背包最大价值为前i-1个物品的最大价值
			else
				dp[i][j] = max(dp[i - 1][j], dp[i - 1][j - weight[i]] + value[i]);
		}
	}
	cout << "背包最大价值:"<<dp[n - 1][bagW];
}

(四)完整代码  

#include<iostream>
#include<vector>
using namespace std;
template<typename T1, typename T2>
void Init(int n,vector<T1>&v,vector<T2>&w)//物品初始化
{
	cout << "依次输入物品的价值和重量:"<<endl;
	for (int i = 0; i < n; i++)
	{
		cin >> v[i] >> w[i];
	}
	//直接初始化物品
	//vector<T1>value{ 1501.6,3000,2000.8 };
	//v = value;
	//vector<T2>weight{ 1,4,3 };
	//w = weight;
}
template<typename T1,typename T2>
void Knapsack(int bagW,int n)
{
	vector<T1>value(n);//存放物品价值
	vector<T2>weight(n);//物品数量
	Init(n,value,weight);//初始化物品信息
	vector<vector<T1>>dp(n,vector<T1>(bagW+1,0));//背包最大价值

	for (int i = 0; i < n; i++)//初始化第一列
	{
		dp[i][0] = 0;
	}
	for (int j = 1; j <= bagW; j++)//初始化第一行
	{
		if (j < weight[0])//背包容量小于物品重量
			dp[0][j] = 0;
		else
			dp[0][j] = value[0];
	}

	for (int i = 1; i < n; i++)//i是物品序号
	{
		for (int j = 1; j <=bagW; j++)//j表示背包容量
		{
			if (j < weight[i])//容量为j的背包装不了物品i
				dp[i][j] = dp[i - 1][j];//此时背包最大价值为前i-1个物品的最大价值
			else
				dp[i][j] = max(dp[i - 1][j], dp[i - 1][j - weight[i]] + value[i]);
		}
	}
	cout << "背包最大价值:"<<dp[n - 1][bagW];
}

int main()
{
	Knapsack<float,int>(4,3);//背包最大容量为4,物品个数为3
	return 0;
}

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

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

相关文章

深入跨域问题(4) - 利用代理解决跨域

在上面的文章中&#xff0c;我们依此了解到&#xff0c;CORS &#xff0c;JSONP 两种方式实现跨域请求。 这两种方法&#xff0c;都需要 前后端进行配合 才能够正确地处理跨域问题。 今天介绍一种方法&#xff0c;不需要前后端配合&#xff0c;前端可独立完成。 众所周知&…

Java的基本语法格式

在编写JavaJavaJava代码过程中&#xff0c;必须先声明一个类&#xff0c;类使用class字眼定义&#xff0c;在class前面可以有一些修饰符&#xff1a; class的修饰符有&#xff1a; public&#xff0c;final(默认是default&#xff0c;只允许在同一包中进行访问) public&#xf…

C++STL-priority_queue的实现

大家好&#xff01;这篇文章&#xff0c;主要讲解一下这个优先级队列&#xff0c;还包含了仿函数等等的知识。希望大家能够一起加油&#xff01;&#xff01;&#xff01; 文章目录1. priority_queue的实现1.1 push函数1.2 pop函数1.3 top函数和empty函数2. 仿函数2.1 仿函数的…

【C进阶】模拟实现atoi函数

⭐博客主页&#xff1a;️CS semi主页 ⭐欢迎关注&#xff1a;点赞收藏留言 ⭐系列专栏&#xff1a;C语言进阶 ⭐代码仓库&#xff1a;C Advanced 家人们更新不易&#xff0c;你们的点赞和关注对我而言十分重要&#xff0c;友友们麻烦多多点赞&#xff0b;关注&#xff0c;你们…

day21-多线程

1.实现多线程 1.1简单了解多线程【理解】 是指从软件或者硬件上实现多个线程并发执行的技术。 具有多线程能力的计算机因有硬件支持而能够在同一时间执行多个线程&#xff0c;提升性能。 1.2并发和并行【理解】 并行&#xff1a;在同一时刻&#xff0c;有多个指令在多个CP…

【数据结构】双链表实现

双链表实现双链表LinkedList的使用ArrayList 和 LinkedList的区别双链表 双链表的结点其实就是在单链表结点的基础上多了一个存储前一个节点地址的域&#xff0c;例如&#xff1a; 接下来就实现双链表的各种操作&#xff0c;首先定义好双链表的结构&#xff1a; public class…

【经典算法】双指针(尺取法):爱,是双向奔赴,还是你追我赶?

&#x1f451;专栏内容&#xff1a;算法学习随笔⛪个人主页&#xff1a;子夜的星的主页&#x1f495;座右铭&#xff1a;日拱一卒&#xff0c;功不唐捐 目录一、前言二、左右指针&#xff08;双向奔赴&#xff09;1、定义2、回文检查三、快慢指针&#xff08;你追我赶&#xff…

将字符串代码编译为字节代码对象 compile()

【小白从小学Python、C、Java】【计算机等级考试500强双证书】【Python-数据分析】将字符串代码编译为字节代码对象compile()[太阳]选择题关于以下python代码表述错误的一项是?sx1y2print("xy",xy)print("【显示】s&#xff1a;")print(s)print("【执…

fpga实操训练(一个典型的fpga系统)

【 声明:版权所有,欢迎转载,请勿用于商业用途。 联系信箱:feixiaoxing @163.com】 对于刚学习fpga的同学来说,很多人可能一开始并不了解,一个典型的fpga系统应该是什么样子的。今天正好来熟悉一下。此外,我们也可以通过这个系统,从另外一个角度学习下,为什么要…

【蓝桥杯】简单数论——快速幂矩阵快速幂

1、快速幂 1.1运算模 定义&#xff1a;模运算为a除以m的余数&#xff0c;记为a mod m&#xff0c;有a mod m a % m。 模运算是大数运算中的常用操作&#xff1a;如果一个数太大&#xff0c;无法直接输出&#xff0c;或者不需要直接输出&#xff0c;可以把它取模后&#xff…

2.4总线操作和定时

文章目录一、引子二、介绍1.总线周期2.总线定时规范三、同步定时方式1.过程2.特点3.优缺点①优点②缺点四、异步定时方式1.介绍2.三种方式&#xff08;1&#xff09;不互锁方式&#xff08;2&#xff09;半互锁方式&#xff08;3&#xff09;全互锁方式3.优缺点①优点②缺点五、…

Allegro如何统计包含过孔长度的网络长度操作指导

Allegro如何统计包含过孔长度的网络长度操作指导 当需要统计网络长度的时候,可以通过element选择nets看到网络的长度,但是当网络换层了,并且需要统计到过孔的长度,类似下图 Allegro可以快速的统计网络的长度,并且包含过孔的长度 具体操作如下 选择Setup选择Constraint –…

设计模式 - 六大设计原则之SRP(单一职责)

文章目录概述CaseBad ImplBetter Impl1. 定义接口2. 职责分离-多种实现类3. 单元测试小结概述 单一职责原则&#xff08;Single Responsibility Principle&#xff0c; SRP&#xff09;又称单一功能原则&#xff0c;是面向对象的五个基本原则&#xff08;SOLID&#xff09;之一…

2022这一年

前言 一年过得好快啊&#xff0c;这个年终总结不知道该写点啥&#xff0c;所以一直到现在也没动笔。 但如果不写吧&#xff0c;总感觉少了点什么。就像过年守夜&#xff0c;反正我是每年都要等到凌晨12点放完鞭炮后才睡。 前些天也看到不少博主发布了2022年终总结&#xff0c;…

【ARM体系结构】之相关概念与公司简介

1、ARM相关的概念 机器码&#xff1a;计算机可以识别的0和1的组合。即高低电平的信号&#xff0c;1高电平信号&#xff0c;0低电平信号 汇编指令&#xff1a;编译器可以将汇编指令&#xff08;存在代码段&#xff09;编译成为机器码&#xff0c;执行汇编指令可以完成相应的汇编…

【进击的算法】基础算法——动态规划

&#x1f37f;本文主题&#xff1a;动态规划 &#x1f388;更多算法&#xff1a;回溯算法 &#x1f495;我的主页&#xff1a;蓝色学者的主页 文章目录一、前言二、概念2.1概念一&#xff1a;状态转移2.2概念二&#xff1a;Dp数组三、例题3.1斐波那契数列3.1.1题目描述3.1.2状态…

JQUERY总结(四)

对象拷贝&#xff1a; <script src"jQuery.min.js"></script> <script>$(function(){// var targetObj{};// var obj{// id:0,// name:"xinyi",// location:"henan"// };// //覆盖以前的相同key值对应的数据// $.…

【自然语言处理】基于NLP的电影评论情感分析模型比较

基于NLP的电影评论情感分析模型比较一段时间以来&#xff0c;使用机器学习的 NLP 任务借助 BERT&#xff08;Bidirectional Encoder Representations from Transformers&#xff09;模型被认为是当前的黄金标准。这些模型通常用于我们日常的许多语言处理任务&#xff0c;比如谷…

Java面试题,线程安全问题

线程安全问题一、对线程安全的理解&#xff08;实际上是内存安全&#xff09;二、Thread类的继承、Runable接口的重写三、守护线程四、ThreadLocal原理和使用场景五、sleep、wait、join、yield六、线程池、解释线程池参数一、对线程安全的理解&#xff08;实际上是内存安全&…

JVM面试一

5. JVM 5.1 JVM包含哪几部分? 参考答案 JVM 主要由四大部分组成:ClassLoader(类加载器),Runtime Data Area(运行时数据区,内存分区),Execution Engine(执行引擎),Native Interface(本地库接口),下图可以大致描述 JVM 的结构。 JVM 是执行 Java 程序的虚拟计算…