动态规划算法与典型例题

article/2025/10/7 3:56:44

目录

前言

一、动态规划要素(条件)

二、动态规划算法设计步骤

三、复杂度分析

四、典型例题1——游艇租聘

五、典型例题2——0-1背包问题

六、典型例题3——跳台阶问题

七、典型例题4——强盗抢劫问题

总结


前言

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


一、动态规划要素(条件)

        1.最优子结构,问题的最优解包括子问题的最优解。如果不具有最优子结构则不能使用动态规划。

        2.子问题重叠,在求解子问题时有大量子问题是重复的,那么只需求解一次,然后把它存储到表格中,以后使用时直接查询,不需重复求解。子问题重叠不是使用动态规划的必要条件,但是只有存在子问题重叠才能彰显动态规划的优势。线

二、动态规划算法设计步骤

        1.分析最优解的结构特征

        2.建立最优值递归式

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

        4.构造最优解

        很多复杂的问题很难找到递归式,但是一旦找到递归式,那么算法已经实现99%。

三、复杂度分析

        时间复杂度:一般为O(n), O(n^2), O(n^3)等n的整数方,看其需要几重循环。

        空间复杂度:一般为O(1),O(n), O(n*m)等,看其子问题最优解与几个控制量有关。

四、典型例题1——游艇租聘

        问题描述:长江游艇俱乐部在长江上设置了n个游艇租聘站,游客可以在这些租聘站租用游艇,然后在下游的任何一个租聘站归还。游艇出租站i到j的租金为r(i, j),1 <=  i< j<= n,设计一个算法,计算从出租站i到j所需的最少租金。

        构造二维数组m[i][j]表示从出租站i到j的最少租金,那么两个子问题:(i, i+1, ... , k), (k, k+1, ..., j)最优值分别是m[i][k]和m[k][j]。

递归式:

m[i][j]=\left\{\begin{matrix} 0 & ,j=i \\ r[i][j] &,j=i+1 \\ min \begin{Bmatrix} m[i][k]+m[k][j], r[i][j] \end{Bmatrix}& ,j > i+1 , i < k < j \end{matrix}\right.

伪代码:

void rent()
{int i, j, k, d;for(d = 3, d <= n; ++d)//将问题分为小规模d。d=0时,租金为0;d=1时,租金为单站租金{for(i = 1; i <= n-d+1; ++i)//起点{j = i+d-1;//终点for(k = i+1; k < i+d-1; ++k)//换乘站,求解每一小规模子问题的解{int temp = m[i][k] + m[k][j];if(temp < m[i][j]) m[i][j] = temp;}}}
}

        总体的求解过程为沿着主对角线一层一层向上,直到右上角填充完毕,则右上角值为所求解。

        时间复杂度O(n^3),空间复杂度O(n^2)。如果在原数组上更新计算值,则空间复杂度可以优化到O(1)。

五、典型例题2——0-1背包问题

        问题描述:动态规划针对不可切割物品,可切割物品可以使用贪心算法。

        约束条件:

\left\{\begin{matrix} \sum w_i*x_i \leqslant W&,1 \leqslant i \leqslant n \\ x_i\in \begin{Bmatrix} 0,1 \end{Bmatrix}& ,1 \leqslant i \leqslant n \end{matrix}\right.

        目标函数:

max\sum v_i*x_i, 1 \leqslant i \leqslant n

        c[i][j]表示前i件物品放入容量为j的购物车可以获得的最大价值。

        递归式:

c[i][j]=\left\{\begin{matrix} c[i-1][j] &, j<w_i \\ max\begin{Bmatrix} c[i-1][j], c[i-1][j-w[i]]+v[i] \end{Bmatrix}& ,j\geqslant w_i \end{matrix}\right.

for(int i = 1; i <= n; ++i)
{for(int j = 1; j <= w; ++j){if(j < w[i])//i物品重量大于购物车,则不放入{c[i][j] = c[i-1][j];}else{//比较i物品放入后能否让购物车价值最大c[i][j] = max(c[i-1][j], c[i-1][j-w[i]]+v[i]);}}
}

        总体求解过程一层一层求解,右下角为所求解。

        时间复杂度O(n*W),空间复杂度O(n*W)。

六、典型例题3——跳台阶问题

        问题描述:有 N 阶楼梯,每次可上一阶或两阶,求有多少种上楼梯的方法。

        容易证明该递推式是斐波拉切数列,

f(n) = n, n < 3

f(n)=f(n-1)+f(n-2), n\geqslant 3

        代码:

int jump(int n)
{if(n < 3) return n;int i = 1, j = 2, res = 0;for(int k = 0; k < n-2; ++k){res = i + j;i = j;j = res;}return res;
}

        时间复杂度O(n),空间复杂度O(1)。

七、典型例题4——强盗抢劫问题

        问题描述:强盗抢劫一排房间,每个房间都有钱,不能抢劫两个相邻的房间,要求抢的钱最多。

        一维数组dp[i]是当抢到第i个数时,能抢到最大值,从局部最大值推到最终结果最大。

        递归式:

DP[i] = max(DP[i-2] + nums [ i ], DP[i-1])

        代码:

int robber(vector<int> &nums)
{int size = nums.size();if(size == 0) return 0;if(size == 1) return nums[0];vector<int> dp(size, 0);dp[0] = nums[0];dp[1] = nums[0] > nums[1] ? nums[0] : nums[1];for(int i = 2; i < size; ++i){dp[i] = max(dp[i-2] + nums[i], dp[i-1]);}return dp[size-1];
}

        时间复杂度O(n),空间复杂度O(n)。


总结

        动态规划最大的优势在可以利用已知得出未知,求解过程中最重要且最困难的是要找出状态转移方程(递推式),同时还需要注意边界条件的设置。上述的经典问题解题思路具有较大参考价值。


http://chatgpt.dhexx.cn/article/1fhoyofr.shtml

相关文章

动态规划问题经典例题

目录 前言一、字符串分割二、三角矩阵的最小路径和三、路径总数四、最小路径和五、背包问题六、 回文串分割七、编辑距离八、不同的子序列 前言 DP&#xff08;Dynamic Programming&#xff09;定义&#xff1a; 动态规划是分治思想的延伸&#xff0c;通俗一点来说就是大事化小…

动态规划经典题目总结

微信公众号 在算法中&#xff0c;动态规划题目算是比较经典的一类题目。在找工作中&#xff0c;不管是笔试&#xff0c;还是面试&#xff0c;我们经常会遇到用动态规划来解决问题的情况&#xff0c;有时候面试官还需要我们现场手写出动态规划解法的代码。因此&#xff0c;在求职…

【动态规划】经典例题

一.动态规划三要素 1.最优子结构 2.状态转移方程 &#xff08;核心&#xff09;&#xff08;一般用打表找出规律&#xff09; 3.边界值 二.背包问题 &#xff08;一.题目&#xff09; 1.1题目描述 现在有一个背包但容量有限&#xff0c;最多只能装m千克宝石!有n个宝石&…

【动态规划专栏】--基础-- 动态规划经典题型

目录 动态规划 动态规划思维&#xff08;基础&#xff09; 状态表示&#xff08;最重要&#xff09; 状态转移方程&#xff08;最难&#xff09; 初始化&#xff08;细节&#xff09; 填表顺序&#xff08;细节&#xff09; 返回值&#xff08;结果&#xff09; 1、第 …

动态规划入门详解 内含12道经典动态规划编程题

动态规划入门详解 一 什么是动态规划&#xff1f;&#xff1f; 算法导论中介绍&#xff0c;动态规划和分治方法类似&#xff0c;都是听过子问题的解来解决原问题。下面说一下这2者之间的分别&#xff0c;分治方法将原问题划分为互不相交的子问题&#xff0c;而后将子问题组合…

【刷题日记】动态规划经典题目

&#x1f600;大家好&#xff0c;我是白晨&#xff0c;一个不是很能熬夜&#x1f62b;&#xff0c;但是也想日更的人✈。如果喜欢这篇文章&#xff0c;点个赞&#x1f44d;&#xff0c;关注一下&#x1f440;白晨吧&#xff01;你的支持就是我最大的动力&#xff01;&#x1f4…

Linux命令-sftp文件传输

搭建SFTP服务详见博文&#xff1a;https://blog.csdn.net/cen50958/article/details/90722874 连接SFTP 可使用&#xff1a;sftp --help 查看SFTP的连接参数 [rootstudy ~]# sftp --help usage: sftp [-1Cv] [-B buffer_size] [-b batchfile] [-F ssh_config] [-o ssh_option…

Linux命令(三):SFTP

目录 1、登录 2、文件上传 3、文件下载 4、删除文件/文件夹 5、实战 1、登录 sftp userip 你要用sftp, 当然得登录到sftp服务器&#xff0c; 在linux的shell中执行上面的命令后&#xff0c; linux shell会提示用户输入密码&#xff0c; 我们就输入password吧。 这样就成功…

Linux常用命令——sftp命令

在线Linux命令查询工具(http://www.lzltool.com/LinuxCommand) sftp 交互式的文件传输程序 补充说明 sftp命令是一款交互式的文件传输程序&#xff0c;命令的运行和使用方式与ftp命令相似&#xff0c;但是&#xff0c;sftp命令对传输的所有信息使用ssh加密&#xff0c;它还…

SFTP命令常用操作

SFTP相关(等价于rz/sz&#xff0c;此方式适用于没有工具的情况下&#xff0c;前提是保证sftp默认端口22开放) lcd 本地文件路径 进入到本地的某个目录下 cd 远程文件路径 进入到远程的某个目录下 lpwd 显示本地的当前目录的路径 pwd 显示远程的当前目录的路径 这里只介绍常…

SFTP基本功之get、put命令操作

简述 在安装好linux系统之后&#xff0c;开始不断安装部署各种工具&#xff0c;其中很多工具版本太老使得无法使用wget下载&#xff0c;而只能用put命令从本地硬盘中上传之linux系统内安装&#xff0c;而当我编写系统克隆mongodb数据库时&#xff0c;又了解到了get命令&#x…

SFTP命令的使用,sftp传文件

背景&#xff1a;从Windows系统向类unix系统传送文件&#xff0c;使用Windows系统自带的SFTP命令进行文件传送(不用下载F开头&#xff0c;X开头的ftp工具) 背景分割线 上干货&#xff1a;1.WinX&#xff0c;按A&#xff0c;输入SFTP root192.168.162.236 回车&#xff1b; &…

SFTP登录及命令行用法

sftp命令行登录过程 ① sftp xxx.xxx.xxx.xxx 登录&#xff08;默认root用户&#xff09;&#xff0c;若指定用户 sftp bluexxx.xxx.xxx.xxx 进行登录&#xff08;blue为用户名&#xff09; ② 登录成功后&#xff0c;会提示输入 密码 ③ 然后&#xff0c;可进入目录&#xf…

SFTP命令用法(上传和下载 )

一、SFTP SFTP是Secure File Transfer Protocol的缩写&#xff0c;安全文件传送协议。可以为传输文件提供一种安全的网络的加密方法。SFTP与FTP有着几乎一样的语法和功能。SFTP为SSH的其中一部分&#xff0c;是一种传输档案至Blogger伺服器的安全方式。其实在SSH软件包中&…

Linux基础命令 sftp命令的使用

SFTP&#xff08;Secure File Transfer Protocol&#xff0c;安全文件传输协议&#xff09;是一种基于可靠数据流&#xff08;data stream&#xff09;&#xff0c;提供文件存取和管理的网络传输协议&#xff0c;与 FTP 协议相比&#xff0c;SFTP 在客户端与服务器间提供了一种…

sftp常用命令介绍

sftp常用命令&#xff1a; 1. sftp 登录sftp服务器 sftp userip ​​​​​​ 如需要看全部命令&#xff1a;则使用help即可 2. pwd和lpwd 、 ls和lls 、cd和lcd 等 sftp登录之后默认操作是远程服务器&#xff0c;当需要操作本地时&#xff0c;就需要在前边加“l”&#…

Linux中使用sftp的常用命令

前言 在数据库远程维护的过程中&#xff0c;经常需要和本机进行数据的交互&#xff0c;常用的交互方式为ftp&#xff0c;但是这种方式需要确保21端口和ftp服务都存在。在远程访问服务器的时候大部分使用ssh来进行连接&#xff0c;其使用的端口为22端口&#xff0c;与之共用的数…

单链表的基本操作-查找

【问题描述】 实现有头结点单链表查找算法&#xff1a;根据关键字值查找其在单链表中的位置(第一次出现的位置)。 【输入形式】 第一行输入整数n&#xff08;n不大于1000&#xff09;&#xff0c;表示单链表长度&#xff1b; 第二行输入若干个整数&#xff08;以非法整数或…

单链表的基本操作(C语言+图解分析)

目录 一、单链表的建立 1、头插法 2、尾插法 二、插入结点操作 三、删除节点操作 四、单链表操作的一些常见问题 1、结构体变量和结构体指针的区别&#xff1f; 2、什么时候要malloc&#xff1f; 3、形参里面出现了取地址符(&)&#xff0c;有什么作用&#xff1f;…

c++单链表的基本操作(全)

俩个基本插入方法 #include <bits/stdc.h> using namespace std; typedef struct LNode { int date; //节点的数据域 struct LNode *next; //节点的指针域 }LNode,*LinkList; // LinkList 为指向结构体LNode的指针类型bool Initlist_L(LinkList &L) …