【算法基础】简单的动态规划!你没见过的全新视角!

2024-05-13 20:36

本文主要是介绍【算法基础】简单的动态规划!你没见过的全新视角!,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

    • 动态规划导论
      • 使用动态规划加速斐波那契数列(记忆化)
      • 自底向上的动态规划
      • 经典的动态规划问题

动态规划导论

动态规划的关键是避免重复的计算。通常情况下,动态规划算法解决的问题可以用递归的方法解决。可以先尝试将问题写出最朴素的递归算法,再使用一个表来保存中间结果,这种属于自底向上的动态规划,或者叫做“记忆化搜索”。

一个最经典的例子就是求斐波那契数列。它的递归式是:当 n ≥ 2 n \geq 2 n2时, f ( n ) = f ( n − 1 ) + f ( n − 2 ) f(n) = f(n - 1) + f(n - 2) f(n)=f(n1)+f(n2) n = 0 n = 0 n=0 f ( 0 ) = 0 f(0) = 0 f(0)=0 n = 1 n = 1 n=1时, f ( 1 ) = 1 f(1) = 1 f(1)=1。用C++来实现就是下面这样:

int f(int n) {if (n == 0) return 0;if (n == 1) return 1;return f(n - 1) + f(n - 2);
}

它的时间复杂度时 O ( 2 n ) O(2^n) O(2n),因为每一个 f ( n ) f(n) f(n)的计算都要调用两次相似的规模的子问题( f ( n − 1 ) f(n-1) f(n1) f ( n − 2 ) f(n-2) f(n2))。

使用动态规划加速斐波那契数列(记忆化)

递归算法需要指数的时间来计算斐波那契数列。这意味着我们只能处理很小的一些输入。比如要计算 f ( 29 ) f(29) f(29)需要超过一百万次函数调用。

为了加快计算速度,我们注意到子问题的规模仅仅是 O ( n ) O(n) O(n)的。就是说,为了计算 f ( n ) f(n) f(n)我们只需要知道$f(n-1),f(n-1),…,f(0) $。因此,相比于重复计算这些子问题,我们通过把这些问题的结果存在一个表中来避免重复的计算。已经计算过的子问题的调用可以通过这个表马上返回结果,避免了指数指数级别的调用次数。

每一次的调用都会先检查表中数据,这会花费 O ( 1 ) O(1) O(1)的时间。如果我们之前已经计算过了,那就返回结果,否则,正常地计算。总共的时间复杂度是 O ( n ) O(n) O(n)。相对于之前的算法来说是一个巨大的提升。

const int MAXN = 100;
bool found[MAXN];
int memo[MAXN];int f(int n) {if (found[n]) return memo[n];if (n == 0) return 0;if (n == 1) return 1;found[n] = true;return memo[n] = f(n - 1) + f(n - 2);
}

使用记忆化搜索的方法, f ( 29 ) f(29) f(29)的计算只需要57次的调用。不过我们还要注意到这是结果的正确性还取决于我们使用的数据类型,在32位整数类型下,我们最多能计算第46位的斐波那契数。

通常情况下我们会在数组中保存着些数,因为在数组中查找的时间是 O ( 1 ) O(1) O(1)。实际上我们可以使用任何我们喜欢的数据结构来保存这些数。比如maps或者unordered_maps。

比如:

unordered_map<int, int> memo;
int f(int n) {if (memo.count(n)) return memo[n];if (n == 0) return 0;if (n == 1) return 1;return memo[n] = f(n - 1) + f(n - 2);
}

或者类似的:

map<int, int> memo;
int f(int n) {if (memo.count(n)) return memo[n];if (n == 0) return 0;if (n == 1) return 1;return memo[n] = f(n - 1) + f(n - 2);
}

这两种保存数据的方式在通常情况下会比数组要慢,但是当状态是向量或者字符串时,这样保存还是很有用的。

最朴素的计算递归算法时间复杂度的方式是:
每一个子问题计算时间 ∗ 子问题个数 每一个子问题计算时间 * 子问题个数 每一个子问题计算时间子问题个数
使用平衡二叉树(C++中的map)来保存状态信息的话,最终需要 O ( n l o g n ) O(nlogn) O(nlogn)的时间因为每次的插入和查找都会需要 O ( l o g n ) O(logn) O(logn)的时间。而子问题的数量一共有 O ( n ) O(n) O(n)个。

上面这种方式叫做自顶向下的,因为我们从询问的数开始计算,并且计算的过程是从上到下的,通过记忆化的方式保留中间的计算结果。

自底向上的动态规划

到目前位置我们只看到了记忆化搜索这种自顶向下的动态规划。其实我们也可以用自底向上的动态规划来解决问题。自底向上方式和自顶向下的方式完全相反,我们从最底层开始(递归的初始条件(base case)),然后扩展到更多的数。

为了实现自底向上的动态规划,我们需要在数组里初始化初始条件。然后在数组中应用递归式:

const int MAXN = 100;
int fib[MAXN];int f(int n) {fib[0] = 0;fib[1] = 1;for (int i = 2; i <= n; i++) fib[i] = fib[i - 1] + fib[i - 2];return fib[n];
}

当然了,这种写法其实有些愚蠢。因为首先如果我们重复调用f(n),会重复计算。其次我们只需要之前的两个数就可以计算当前的值。因此我们把空间复杂度从 O ( n ) O(n) O(n)减少到 O ( 1 ) O(1) O(1)

代码:

const int MAX_SAVE = 3;
int fib[MAX_SAVE];int f(int n) {fib[0] = 0;fib[1] = 1;for (int i = 2; i <= n; i++)fib[i % MAX_SAVE] = fib[(i - 1) % MAX_SAVE] + fib[(i - 2) % MAX_SAVE];return fib[n % MAX_SAVE];
}

注意到把MAXN变成了MAX_SAVE。这是因为我们需要访问的数其实只有3个。需要的空间和输入无关,根据定义,这就是 O ( 1 ) O(1) O(1)的空间复杂度。并且在这段代码中还使用了一个常用的小技巧(使用模运算)来维护我们需要的数。

上面就是动态规划的基础:不要重复你之前的工作。

一个更好掌握动态规划的方式是学习一些经典的例子。

经典的动态规划问题

  • 0-1 Knapsack
  • Subset Sum
  • Longest Increasing Subsequence
  • Counting all possible paths from top left to bottom right corner of a matrix
  • Longest Common Subsequence
  • Longest Path in a Directed Acyclic Graph (DAG)
  • Coin Change
  • Longest Palindromic Subsequence
  • Rod Cutting
  • Edit Distance
  • Bitmask Dynamic Programming
  • Digit Dynamic Programming
  • Dynamic Programming on Trees

这篇关于【算法基础】简单的动态规划!你没见过的全新视角!的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/986759

相关文章

RedHat运维-Linux文本操作基础-AWK进阶

你不用整理,跟着敲一遍,有个印象,然后把它保存到本地,以后要用再去看,如果有了新东西,你自个再添加。这是我参考牛客上的shell编程专项题,只不过换成了问答的方式而已。不用背,就算是我自己亲自敲,我现在好多也记不住。 1. 输出nowcoder.txt文件第5行的内容 2. 输出nowcoder.txt文件第6行的内容 3. 输出nowcoder.txt文件第7行的内容 4. 输出nowcode

一份LLM资源清单围观技术大佬的日常;手把手教你在美国搭建「百万卡」AI数据中心;为啥大模型做不好简单的数学计算? | ShowMeAI日报

👀日报&周刊合集 | 🎡ShowMeAI官网 | 🧡 点赞关注评论拜托啦! 1. 为啥大模型做不好简单的数学计算?从大模型高考数学成绩不及格说起 司南评测体系 OpenCompass 选取 7 个大模型 (6 个开源模型+ GPT-4o),组织参与了 2024 年高考「新课标I卷」的语文、数学、英语考试,然后由经验丰富的判卷老师评判得分。 结果如上图所

Vim使用基础篇

本文内容大部分来自 vimtutor,自带的教程的总结。在终端输入vimtutor 即可进入教程。 先总结一下,然后再分别介绍正常模式,插入模式,和可视模式三种模式下的命令。 目录 看完以后的汇总 1.正常模式(Normal模式) 1.移动光标 2.删除 3.【:】输入符 4.撤销 5.替换 6.重复命令【. ; ,】 7.复制粘贴 8.缩进 2.插入模式 INSERT

零基础STM32单片机编程入门(一)初识STM32单片机

文章目录 一.概要二.单片机型号命名规则三.STM32F103系统架构四.STM32F103C8T6单片机启动流程五.STM32F103C8T6单片机主要外设资源六.编程过程中芯片数据手册的作用1.单片机外设资源情况2.STM32单片机内部框图3.STM32单片机管脚图4.STM32单片机每个管脚可配功能5.单片机功耗数据6.FALSH编程时间,擦写次数7.I/O高低电平电压表格8.外设接口

大语言模型(LLMs)能够进行推理和规划吗?

大语言模型(LLMs),基本上是经过强化训练的 n-gram 模型,它们在网络规模的语言语料库(实际上,可以说是我们文明的知识库)上进行了训练,展现出了一种超乎预期的语言行为,引发了我们的广泛关注。从训练和操作的角度来看,LLMs 可以被认为是一种巨大的、非真实的记忆库,相当于为我们所有人提供了一个外部的系统 1(见图 1)。然而,它们表面上的多功能性让许多研究者好奇,这些模型是否也能在通常需要系

ps基础入门

1.基础      1.1新建文件      1.2创建指定形状      1.4移动工具          1.41移动画布中的任意元素          1.42移动画布          1.43修改画布大小          1.44修改图像大小      1.5框选工具      1.6矩形工具      1.7图层          1.71图层颜色修改          1

代码随想录算法训练营:12/60

非科班学习算法day12 | LeetCode150:逆波兰表达式 ,Leetcode239: 滑动窗口最大值  目录 介绍 一、基础概念补充: 1.c++字符串转为数字 1. std::stoi, std::stol, std::stoll, std::stoul, std::stoull(最常用) 2. std::stringstream 3. std::atoi, std

回调的简单理解

之前一直不太明白回调的用法,现在简单的理解下 就按这张slidingmenu来说,主界面为Activity界面,而旁边的菜单为fragment界面。1.现在通过主界面的slidingmenu按钮来点开旁边的菜单功能并且选中”区县“选项(到这里就可以理解为A类调用B类里面的c方法)。2.通过触发“区县”的选项使得主界面跳转到“区县”相关的新闻列表界面中(到这里就可以理解为B类调用A类中的d方法

人工智能机器学习算法总结神经网络算法(前向及反向传播)

1.定义,意义和优缺点 定义: 神经网络算法是一种模仿人类大脑神经元之间连接方式的机器学习算法。通过多层神经元的组合和激活函数的非线性转换,神经网络能够学习数据的特征和模式,实现对复杂数据的建模和预测。(我们可以借助人类的神经元模型来更好的帮助我们理解该算法的本质,不过这里需要说明的是,虽然名字是神经网络,并且结构等等也是借鉴了神经网络,但其原型以及算法本质上还和生物层面的神经网络运行原理存在

自制的浏览器主页,可以是最简单的桌面应用,可以把它当成备忘录桌面应用

自制的浏览器主页,可以是最简单的桌面应用,可以把它当成备忘录桌面应用。如果你看不懂,请留言。 完整代码: <!DOCTYPE html><html lang="zh-CN"><head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><ti