提醒:本文需读者具有以下C/C++基础:
- 循环结构,数组
- 函数
- 递归
普通人和小白可以滚了,否则待会脑袋爆掉医药费我可不报销。 本人主要活动于知乎,发文顺序:知乎——>掘金,知乎主页:lsely - 知乎 (zhihu.com)
动态规划的定义
动态规划(Dynamic Programming,简称DP),是一种通过把复杂问题分解为相对简单的子问题的方式求解最优策略的方法,可以有效的解决背包问题、资源分配问题、最短路径问题和复杂系统可靠性问题等求最优解策略问题。
dynamic programming is a method for solving a complex problem by breaking it down into a collection of simpler subproblems. ——Wikipedia
译文:一般这些子问题很相似,可以通过函数关系式递推出来。然后呢,动态规划就致力于解决每个子问题一次,减少重复计算,比如斐波那契数列就可以看做入门级的经典动态规划问题。 ——维基百科
说实话,我看我也头痛,拿来装杯不错,真理解还得靠自己啊!我给大家翻译翻译什么叫动态规划:
动态规划,别看名字很高深,其实本质就是将复杂的问题化为简单的子问题,再将子问题化为更为简单的子问题,直至简单到不能再简单,一下就能给出答案的程度。就酱,很复杂的题就迎刃而解了。还有动态规划Plus版,就是将每一次子问题的答案保存下来,这样以后要用到时直接调用就行了,效率更上十层楼。也就是记忆化搜索。
什么?还不能理解?那就看这几个小剧场:
A:1+2+3+4+5+6+7=28,
那么1+2+3+4+5+6+7+8等于几?
B:36。
A:你是怎么算出来的?
B:1+2+3+4+5+6+7=28,那么1+2+3+4+5+6+7+8=28+8=36.
这就是记忆化搜索。
核心思想
我个人将其总结为一型三征,一型就是动态规划适合解决的问题的模型,
三征也就是三个特征:最优子结构,无后效性,记忆化搜索。 由于过于抽象,以下内容不要求全部背下,但一定要能够说出这三个性质中粗体部分。
1.最优子结构
如果问题的最优解所包含的子问题的解也是最优的,我们就称该问题具有最优子结构性质(即满足最优化原理)。最优子结构性质为动态规划算法解决问题提供了重要线索。
2.无后效性
即子问题的解一旦确定,就不再改变,不受在这之后、包含它的更大的问题的求解决策影响。
3.记忆化搜索
也称子问题重叠性质,子问题重叠性质是指在用递归算法自顶向下对问题进行求解时,每次产生的子问题并不总是新问题,有些子问题会被重复计算多次。动态规划算法正是利用了这种子问题的重叠性质,每次计算完一次子问题的答案,再将答案保存下来,在遇到重复的自问题时直接调用之前保存的答案,从而获得更高的效率。
解题步骤
综上所述,那么基本的动态规划题目的解题步骤大体就是酱:
1. 寻找最优子结构(状态表示)
2. 写出状态转移方程(状态计算)
3. 边界初始化
光说不练是假把式,我们先来看一道力扣极为经典的题目——斐波那契数列。
509.斐波那契数列
某网友:啊,这个我熟!肥波纳粹数列!
我:emm..... 滚去练打字!
本题链接: 509. 斐波那契数 - 力扣(LeetCode)
题目描述
斐波那契数 (通常用 F(n) 表示)形成的序列称为 斐波那契数列 。该数列由 0 和 1 开始,后面的每一项数字都是前面两项数字的和。也就是:
F(0) = 0,F(1) = 1
F(n) = F(n - 1) + F(n - 2),其中 n > 1
给定 n ,请计算 F(n) 。
哎呀,这时候可能有人要说了:
欸嘿嘿,这不简单,直接一个递归不就行了吗
于是他兴冲冲地写完代码:
class Solution {
public:
int fib(int n) {
if(n==0||n==1) return n;
return fib(n-1)+fib(n-2);
}
};
复制粘贴提交:
可以看到勉强虽然但是差不多通过了,但用时极其夸张,那这是为什么呢?别急,我们先来分析一下这个代码的运算过程。
代码分析
我们设 n=5 时,来看一下程序的运算过程:
本人随手画的图,有点简陋,凑活着看吧
欸,你发现了什么?
fib (3)在计算过程中计算了两次
没错,上图中圈画的fib (3)在计算过程中计算了两次,而当我们把 n 设的更大,比如说 n=7。
会发现重复计算已经不是一次两次了,甚至是整个分支都存在重复。而计算越庞大,比如 n=1000,10000,重复计算也就越多,时间复杂度就十分恐怖了, 要不是力扣大大好,题目中设定了 0<=n<=30,也就是不可能超过30,这样做法肯定会超时。
既然存在这么大的问题,那要怎么优化呢?
优化方案
没错,记忆化搜索,我们在前面讲过了,当子问题存在大量重复计算时,利用子问题重叠性质,就可以将子问题的答案记录下来,在需要用到时就直接调用,一劳永逸,很轻松就解决了重复计算的问题。
class Solution {
public:
int fib(int n) {
if (n < 2) {
return n;
}
int p = 0, q = 0, r = 1;
for (int i = 2; i <= n; ++i) {
p = q;
q = r;
r = p + q;
}
return r;
}
};
提交上去:
效率完爆递归算法。
练习题单
做完这道题,是不是觉得自己又"行"了?
那我可得好好挫挫你的锐气,动态规划难题多的是!来,收下这份大礼包(难度依次增加):
- 70. 爬楼梯 - 力扣(LeetCode)
- 746. 使用最小花费爬楼梯 - 力扣(LeetCode)
- Loading Question... - 力扣(LeetCode)
- 5. 最长回文子串 - 力扣(LeetCode)
- 53. 最大子数组和 - 力扣(LeetCode)
- 95. 不同的二叉搜索树 II - 力扣(LeetCode)
- 213. 打家劫舍 II - 力扣(LeetCode)
- 32. 最长有效括号 - 力扣(LeetCode)
- 123. 买卖股票的最佳时机 III - 力扣(LeetCode)
- 689. 三个无重叠子数组的最大和 - 力扣(LeetCode)
——————本文完结撒花——————