当前位置: 首页 > news >正文

动态规划入门和应用示例

文章目录

  • 前言
  • 斐波那契数列
  • 爬楼梯
  • 总结
    • 优点:
    • 缺点:


前言

动态规划(Dynamic Programming,DP)是运筹学的一个分支,是求解决策过程最优化的数学方法。它主要用于解决一类具有重叠子问题和最优子结构性质的问题。通过把原问题分解为相对简单的子问题的方式,动态规划可以求得复杂问题的最优解。

动态规划的基本思想是将一个复杂的问题分解为若干个相对简单的子问题,通过求解这些子问题,并将它们的解存储起来,以便在求解更大的问题时能够重复利用这些解,从而避免大量的重复计算,提高算法的效率。

以下是动态规划入门的几个关键要点:

  1. 核心思想
  • 动态规划的核心思想是利用过去的数据解决现在的问题。通过分解原问题为相互重叠的子问题,并解决这些子问题来得到原问题的解。
  • 动态规划的关键在于“状态”和“状态转移方程”。状态表示问题的某个阶段,而状态转移方程描述了从一个状态转移到另一个状态的规则。
  1. 基本步骤
  • 定义状态:明确问题中的状态,即子问题的表示方式。
  • 状态转移方程:建立状态之间的关系,即如何从当前状态转移到下一个状态。
  • 初始化:为状态的初始值设定合理的默认值。
  • 计算顺序:按照某种顺序(通常是自底向上或自左向右)计算所有状态的值。
  • 返回值:返回最终状态的值作为问题的解。
  1. 应用实例
  • 斐波那契数列:经典的动态规划问题,可以通过存储已计算的斐波那契数来避免重复计算。
  • 背包问题:给定一组物品和背包容量,如何选择物品使得背包内物品的总价值最大。
  • 最短路径问题:在图论中,求从一个顶点到另一个顶点的最短路径。
  • 股票买卖问题:计算在给定的股票价格序列中,通过买入和卖出股票能够获得的最大利润。
  1. 实践建议
  • 从简单的问题开始,逐步增加难度,逐步熟悉和掌握动态规划的思想和方法。
  • 多做练习,尤其是经典问题的练习,有助于深入理解动态规划的应用和技巧。
  • 结合实际问题,思考如何将其转化为动态规划问题,并设计合适的状态和状态转移方程。

总之,动态规划是一种强大而灵活的数学工具,适用于求解各种优化问题。通过不断学习和实践,你可以逐渐掌握这一技术并应用于实际问题中。


斐波那契数列

斐波那契数列是一个常见的动态规划问题,其定义为:每个数是前两个数之和,序列的开始两个数是0和1。例如,斐波那契数列的前几个数是:0, 1, 1, 2, 3, 5, 8, 13, 21, …

在C++中,我们可以使用动态规划来高效地计算斐波那契数列中的数。下面是一个简单的C++代码示例,它使用动态规划来计算第n个斐波那契数:

#include <bits/stdc++.h>int fibonacci(int n) {if (n <= 1) {return n;}std::vector<int> dp(n + 1);dp[0] = 0;dp[1] = 1;for (int i = 2; i <= n; ++i) {dp[i] = dp[i - 1] + dp[i - 2];}return dp[n];
}int main() {int n;std::cout << "Enter the position in the Fibonacci sequence: ";std::cin >> n;std::cout << "Fibonacci number at position " << n << " is: " << fibonacci(n) << std::endl;return 0;
}

在这个代码中,我们创建了一个dp数组来存储斐波那契数列的每一项。数组的索引表示斐波那契数列中的位置,数组的值表示对应位置的斐波那契数。我们从位置2开始迭代,并使用前两项的值来计算当前项的值。最后,我们返回dp[n],即第n个斐波那契数。

注意,这个实现方法对于较大的n值可能不是最高效的,因为它使用了一个大小为n+1的数组来存储中间结果。对于更大的n值,我们可以考虑使用迭代方法而不是动态规划,因为迭代方法只需要存储前两项的值,而不是整个数组。下面是一个迭代方法的示例:

int fibonacci(int n) {if (n <= 1) {return n;}int a = 0, b = 1, temp;for (int i = 2; i <= n; ++i) {temp = a + b;a = b;b = temp;}return b;
}

在这个迭代版本中,我们只使用了三个变量abtemp来依次计算斐波那契数列中的每一项,而不是使用整个数组。这种方法在内存使用上更加高效,特别是对于非常大的n值。

爬楼梯

C++ 动态规划入门的一个经典例子就是“爬楼梯”问题。这个问题描述如下:

假设你正在爬楼梯,需要 n 阶你才能到达楼顶。每次你都可以爬 1 或 2 个台阶。你有多少种不同的方法可以爬到楼顶呢?

为了解决这个问题,我们可以使用动态规划。我们可以定义一个数组 dp,其中 dp[i] 表示到达第 i 阶楼梯的方法数。根据题目,我们知道:

  • dp[0 = 1(只有1种方法到达第1阶楼梯,即爬1个台阶)
  • dp[1] = 2(有2种方法到达第2阶楼梯,即爬1个台阶两次或爬2个台阶一次)

对于 i >= 2 的情况,到达第 i 阶楼梯的方法数等于到达第 i-1 阶楼梯的方法数(再爬1个台阶)加上到达第 i-2 阶楼梯的方法数(再爬2个台阶)。因此,我们可以得到状态转移方程:

dp[i] = dp[i-1] + dp[i-2]

下面是一个使用 C++ 实现的例子:

#include <bits/stdc++.h>int climbStairs(int m) {if (m <= 2) {return n;}//定义m级台阶走法数组int dp[m];//一级台阶1种走法dp[0] = 1;  //二级台阶2种走法dp[1] = 2;  //第i级台阶走法=第i-1级台阶走法+第i-2级台阶走法for (int i = 2; i < m; ++i) {  dp[i] = dp[i - 1] + dp[i - 2];  }  return dp[m-1];
}int main() {int n;std::cout << "Enter the number of stairs: ";std::cin >> n;std::cout << "Number of ways to climb " << n << " stairs: " << climbStairs(n) << std::endl;return 0;
}

总结

以下是动态规划的主要优缺点:

优点:

  1. 高效性:动态规划通过存储子问题的解,避免了重复计算。对于具有大量重叠子问题的情况,动态规划可以显著减少计算量,提高算法效率。

  2. 全局最优解:动态规划通过自底向上的方式,逐步构建问题的解,保证了最终得到的是全局最优解。在解决优化问题时,动态规划是一种非常有效的工具。

  3. 结构清晰:动态规划通常将问题分解为一系列相互关联的子问题,并按照一定的顺序逐步求解。这种分解方式使得问题的结构更加清晰,便于理解和实现。

  4. 适用范围广:动态规划可以应用于多种类型的问题,包括背包问题、最长公共子序列、最短路径问题等。通过合理的状态定义和状态转移方程设计,动态规划可以很好地解决这些问题。

缺点:

  1. 空间复杂度较高:动态规划通常需要存储大量的子问题解,以便在后续计算中重复利用。这可能导致较高的空间复杂度,特别是在处理大规模问题时,可能需要消耗大量的内存空间。

  2. 设计难度较大:动态规划的核心在于状态的定义和状态转移方程的设计。对于复杂的问题,如何选择合适的状态和状态转移方程可能是一个挑战。设计不当可能导致算法的正确性无法保证或效率较低。

  3. 问题依赖性强:动态规划通常针对特定类型的问题进行设计,对于不同类型的问题可能需要采用不同的策略和方法。这使得动态规划具有一定的局限性,不适用于所有类型的问题。

  4. 可能不是最优解法:虽然动态规划通常能够得到全局最优解,但在某些情况下,可能存在其他更高效的算法或启发式方法来解决相同的问题。因此,在选择算法时需要根据问题的特点进行权衡和比较。

综上所述,动态规划具有高效性和全局最优解的优点,但也可能存在空间复杂度较高和设计难度较大的缺点。在实际应用中,需要根据问题的特点选择合适的算法,并充分利用动态规划的优势来解决实际问题。

相关文章:

动态规划入门和应用示例

文章目录 前言斐波那契数列爬楼梯总结优点&#xff1a;缺点&#xff1a; 前言 动态规划&#xff08;Dynamic Programming&#xff0c;DP&#xff09;是运筹学的一个分支&#xff0c;是求解决策过程最优化的数学方法。它主要用于解决一类具有重叠子问题和最优子结构性质的问题。…...

【C语言】精品练习题

目录 题目一&#xff1a; 题目二&#xff1a; 题目三&#xff1a; 题目四&#xff1a; 题目五&#xff1a; 题目六&#xff1a; 题目七&#xff1a; 题目八&#xff1a; 题目九&#xff1a; 题目十&#xff1a; 题目十一&#xff1a; 题目十二&#xff1a; 题目十…...

数据库(MySQL)—— DML语句

数据库&#xff08;MySQL&#xff09;—— DML语句 什么是DML语句添加数据给全部字段添加数据批量添加数据 修改数据删除数据 什么是DML语句 在MySQL中&#xff0c;DML&#xff08;Data Manipulation Language&#xff0c;数据操纵语言&#xff09;语句主要用于对数据库中的数…...

【最大公约数 并集查找 调和级数】1998. 数组的最大公因数排序

本文涉及知识点 最大公约数 并集查找 调和级数 LeetCode1998. 数组的最大公因数排序 给你一个整数数组 nums &#xff0c;你可以在 nums 上执行下述操作 任意次 &#xff1a; 如果 gcd(nums[i], nums[j]) > 1 &#xff0c;交换 nums[i] 和 nums[j] 的位置。其中 gcd(nums…...

iOS实现一个高性能的跑马灯

效果图 该跑马灯完全通过CATextLayer 实现&#xff0c;轻量级&#xff0c;并且通过 系统的位移动画实现滚动效果&#xff0c;避免了使用displaylink造成的性能瓶颈&#xff0c;使用系统动画&#xff0c;系统自动做了很多性能优化&#xff0c;实现更好的性能&#xff0c;并使用…...

MySQL的视图、存储过程、触发器

视图 介绍 视图是一种虚拟存在的表。视图中的数据并不在数据库中实际存在&#xff0c;行和列数据来自定义视图的查询中使用的表&#xff0c;并且是在使用视图时动态生成的。通俗的讲&#xff0c;视图只保存了查询的SQL逻辑&#xff0c;不保存查询结果。所以我们在创建视图的时…...

【图像特征点匹配】

图像特征点匹配 图像特征点匹配是计算机视觉中的一项关键技术,它涉及在两个或多个图像之间寻找并匹配具有独特属性的点,这些点被称为特征点。 立体视觉:通过匹配同一场景的不同视角图像中的特征点,可以重建场景的三维结构。物体识别:通过匹配物体表面的特征点,可以识别和…...

GZIPOutputStream JSON压缩

一、背景 小王瞥了一眼历史记录表&#xff0c;不禁惊呼&#xff1a;“这表怎么这么大&#xff1f;”同事们闻声纷纷围拢过来查看。仔细一瞧&#xff0c;发现这个表的大小竟然超过了3G。主管随即指示小王打开相应的表数据检查&#xff0c;发现其中存储了用户的权限信息&#xf…...

毫米波雷达原理(含代码)(含ARS548 4D毫米波雷达数据demo和可视化视频)

毫米波雷达原理 1. 传统毫米波雷达1.1 雷达工作原理1.2 单目标距离估计1.3 单目标速度估计1.4 单目标角度估计1.5 多目标距离估计1.6 多目标速度估计1.7多目标角度估计1.7 总结 3. FMCW雷达数据处理算法4. 毫米波雷达的目标解析(含python代码)5. ARS548 4D毫米波雷达数据demo(含…...

3.1 Gateway之路由请求和转发

1.依赖坐标 <!--网关--><dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-gateway</artifactId></dependency><!--服务注册和发现--><dependency><groupId>com.alibab…...

人脸识别开源算法库和开源数据库

目录 1. 人脸识别开源算法库 1.1 OpenCV人脸识别模块 1.2 Dlib人脸识别模块 1.3 SeetaFace6 1.4 DeepFace 1.5 InsightFace 2. 人脸识别开源数据库 2.1 CelebA 2.2 LFW 2.3 MegaFace 2.4 Glint360K 2.5 WebFace260M 人脸识别 (Face Recognition) 是一种基于人的面部…...

Excel 中用于在一个范围中查找特定的值,并返回同一行中指定列的值 顺序不一样 可以处理吗

一、需求 Excel 中&#xff0c;在一列&#xff08;某范围内&#xff09;查找另一列特定的值&#xff0c;并返回同一行中另一指定列的值&#xff0c; 查找列和返回列的顺序不一样 二、 实现 1、下面是一个使用 INDEX 和 MATCH 函数的例子&#xff1a; 假设你有以下数据&…...

MySql-日期分组

一、分别统计各时间各类型数据条数 数据库的 request_time字段 数据类型&#xff1a;timestamp 默认值&#xff1a;CURRENT_TIMESTAMP 例子&#xff1a; 2024-01-26 08:25:48 原数据&#xff1a; 1、将数据按照日期&#xff08;年月日&#xff09;形式输出 按照request_…...

有哪些方法可以在运行时动态生成一个Java类?

使用 Java 反射 API&#x1f6a9;&#xff1a; Java 的反射 API 允许在运行时查询和操作类和对象。虽然反射 API 本身不直接提供生成新类的功能&#xff0c;但可以用于动态调用构造函数、方法和访问字段&#xff0c;这在某些情况下可以作为动态生成类的一部分。 字节码操作库&…...

JAVA两个线程交替打印实现

方案1 Semaphore 机制 通过信息号机制来 协调两个线程&#xff0c;一个线程打印后&#xff0c;给另一个线程释放一个信号量 Semaphore semaphorea new Semaphore(1);Semaphore semaphoreb new Semaphore(0);Thread threada new Thread(new Runnable() {Overridepublic void…...

【C语言】学习C语言

C语言简介 C语言是一门十分流行的编程语言&#xff0c;由美国贝尔实验室的 Dennis Ritchie 在 20 世纪 70 年代开发。 C语言具有高效、可移植、灵活、简单等特点&#xff0c;被广泛应用于操作系统、编译器、数据库、图形界面、嵌入式系统、网络通信、游戏等领域。 本文将带你…...

C 深入指针(2)

目录 1 野指针 1.1 成因 1.2 如何规避野指针 2 assert 断言 2.1 用法 2.2 assert 的优点 2.1 assert 的缺点 3 小注解 3.1 Debug 和 Release 1 野指针 【概念】&#xff1a; 野指针就是指针指向的位置是不可知的&#xff08;随机的、不正确的、没有明确限制的&#…...

FileLink跨网文件交换,推动企业高效协作|半导体行业解决方案

随着信息技术的迅猛发展&#xff0c;全球信息产业已经迎来了前所未有的繁荣与变革。在这场科技革命中&#xff0c;半导体作为信息产业的基础与核心&#xff0c;其重要性日益凸显&#xff0c;半导体的应用场景和市场需求将进一步扩大。 然而&#xff0c;在这一繁荣的背后&#x…...

代码随想录day56 | 动态规划P16 | ● 583. ● 72. ● 编辑距离总结篇

583. 两个字符串的删除操作 给定两个单词 word1 和 word2 &#xff0c;返回使得 word1 和 word2 相同所需的最小步数。 每步 可以删除任意一个字符串中的一个字符。 示例 1&#xff1a; 输入: word1 "sea", word2 "eat" 输出: 2 解释: 第一步将 &quo…...

ASP.NET网络在线考试系统

摘 要 随着计算机技术的发展和互联网时代的到来&#xff0c;人们已经进入了信息时代&#xff0c;也有人称为数字化时代。数在数字化的网络环境下&#xff0c;学生希望得到个性化的满足&#xff0c;根据自己的情况进行学习&#xff0c;同时也希望能够得到科学的评价&#xff0c…...

CTF show Web 红包题第六弹

提示 1.不是SQL注入 2.需要找关键源码 思路 进入页面发现是一个登录框&#xff0c;很难让人不联想到SQL注入&#xff0c;但提示都说了不是SQL注入&#xff0c;所以就不往这方面想了 ​ 先查看一下网页源码&#xff0c;发现一段JavaScript代码&#xff0c;有一个关键类ctfs…...

python打卡day49

知识点回顾&#xff1a; 通道注意力模块复习空间注意力模块CBAM的定义 作业&#xff1a;尝试对今天的模型检查参数数目&#xff0c;并用tensorboard查看训练过程 import torch import torch.nn as nn# 定义通道注意力 class ChannelAttention(nn.Module):def __init__(self,…...

大数据零基础学习day1之环境准备和大数据初步理解

学习大数据会使用到多台Linux服务器。 一、环境准备 1、VMware 基于VMware构建Linux虚拟机 是大数据从业者或者IT从业者的必备技能之一也是成本低廉的方案 所以VMware虚拟机方案是必须要学习的。 &#xff08;1&#xff09;设置网关 打开VMware虚拟机&#xff0c;点击编辑…...

为什么需要建设工程项目管理?工程项目管理有哪些亮点功能?

在建筑行业&#xff0c;项目管理的重要性不言而喻。随着工程规模的扩大、技术复杂度的提升&#xff0c;传统的管理模式已经难以满足现代工程的需求。过去&#xff0c;许多企业依赖手工记录、口头沟通和分散的信息管理&#xff0c;导致效率低下、成本失控、风险频发。例如&#…...

【磁盘】每天掌握一个Linux命令 - iostat

目录 【磁盘】每天掌握一个Linux命令 - iostat工具概述安装方式核心功能基础用法进阶操作实战案例面试题场景生产场景 注意事项 【磁盘】每天掌握一个Linux命令 - iostat 工具概述 iostat&#xff08;I/O Statistics&#xff09;是Linux系统下用于监视系统输入输出设备和CPU使…...

反射获取方法和属性

Java反射获取方法 在Java中&#xff0c;反射&#xff08;Reflection&#xff09;是一种强大的机制&#xff0c;允许程序在运行时访问和操作类的内部属性和方法。通过反射&#xff0c;可以动态地创建对象、调用方法、改变属性值&#xff0c;这在很多Java框架中如Spring和Hiberna…...

算法笔记2

1.字符串拼接最好用StringBuilder&#xff0c;不用String 2.创建List<>类型的数组并创建内存 List arr[] new ArrayList[26]; Arrays.setAll(arr, i -> new ArrayList<>()); 3.去掉首尾空格...

c# 局部函数 定义、功能与示例

C# 局部函数&#xff1a;定义、功能与示例 1. 定义与功能 局部函数&#xff08;Local Function&#xff09;是嵌套在另一个方法内部的私有方法&#xff0c;仅在包含它的方法内可见。 • 作用&#xff1a;封装仅用于当前方法的逻辑&#xff0c;避免污染类作用域&#xff0c;提升…...

Axure 下拉框联动

实现选省、选完省之后选对应省份下的市区...

热烈祝贺埃文科技正式加入可信数据空间发展联盟

2025年4月29日&#xff0c;在福州举办的第八届数字中国建设峰会“可信数据空间分论坛”上&#xff0c;可信数据空间发展联盟正式宣告成立。国家数据局党组书记、局长刘烈宏出席并致辞&#xff0c;强调该联盟是推进全国一体化数据市场建设的关键抓手。 郑州埃文科技有限公司&am…...