从爬楼梯到算法世界:动态规划与斐波那契的奇妙邂逅
从爬楼梯到算法世界:动态规划与斐波那契的奇妙邂逅
在算法学习的旅程中,总有一些经典问题让人印象深刻。“爬楼梯问题”就是其中之一,看似简单的题目,却蕴藏了动态规划与斐波那契数列的深刻联系。今天,我就以这个问题为切入点,带大家深入探讨如何通过动态规划解锁更广阔的算法世界。
1. 问题描述与分析
问题描述
假设你正在爬一座楼梯,共有n
阶,每次你可以选择爬1阶或2阶。问:到达顶楼有多少种不同的爬法?
举个例子:
- 如果有3阶楼梯,你的可能路径是:
- 1阶+1阶+1阶
- 1阶+2阶
- 2阶+1阶
总共有3种爬法。
分析问题
这个问题本质上是求解一个递归关系:
- 如果你站在第
n
阶,那么你是从n-1
阶迈1步或从n-2
阶迈2步到达的。 - 因此,第
n
阶的爬法数目可表示为:
f ( n ) = f ( n − 1 ) + f ( n − 2 ) f(n) = f(n-1) + f(n-2) f(n)=f(n−1)+f(n−2)
初始条件:
- 如果楼梯只有1阶,只有一种爬法: f ( 1 ) = 1 f(1) = 1 f(1)=1
- 如果楼梯有2阶,有两种爬法: f ( 2 ) = 2 f(2) = 2 f(2)=2
聪明的你可能已经发现了,这正是斐波那契数列的定义公式!
2. 解法探索
我们来看看不同的解法,从递归到动态规划,再到优化。以代码为例,每种解法都蕴含了对问题不同层次的思考。
方法一:递归解法(Naive Approach)
这是最直观的实现,但存在巨大的性能瓶颈。
def climb_stairs_recursive(n):"""递归解法,计算爬楼梯的方法数"""if n <= 2:return n # 边界情况return climb_stairs_recursive(n - 1) + climb_stairs_recursive(n - 2)
问题:
- 这种方法的时间复杂度是指数级的( O ( 2 n ) O(2^n) O(2n)),因为存在大量重复计算。
- 比如计算
f(5)
时,需要重复计算f(4)
和f(3)
,效率极低。
方法二:记忆化递归(优化递归)
为了解决重复计算的问题,我们可以利用一个数组或字典来缓存中间结果。
def climb_stairs_memoization(n, memo={}):"""记忆化递归,减少重复计算"""if n in memo:return memo[n]if n <= 2:return n # 边界情况memo[n] = climb_stairs_memoization(n - 1, memo) + climb_stairs_memoization(n - 2, memo)return memo[n]
优势:
- 通过缓存中间结果,将时间复杂度降低到了 O ( n ) O(n) O(n)。
- 但仍然使用了递归,函数栈会占用一定的内存。
方法三:动态规划(Bottom-Up Approach)
动态规划是一种自底向上的方法,通过数组存储每一阶的爬法数,避免递归的栈开销。
def climb_stairs_dp(n):"""动态规划解法"""if n <= 2:return n # 边界情况dp = [0] * (n + 1) # 初始化DP数组dp[1], dp[2] = 1, 2 # 初始条件for i in range(3, n + 1): # 逐步计算每一阶的爬法数dp[i] = dp[i - 1] + dp[i - 2]return dp[n]
优势:
- 时间复杂度是 O ( n ) O(n) O(n)。
- 空间复杂度是 O ( n ) O(n) O(n),因为需要额外数组存储每一阶的结果。
方法四:动态规划的空间优化
进一步优化空间,利用滚动变量替代数组。毕竟我们每次只需要用到前两阶的结果。
def climb_stairs_optimized(n):"""动态规划 + 空间优化"""if n <= 2:return n # 边界情况prev1, prev2 = 1, 2 # 初始化前两阶的爬法数for _ in range(3, n + 1): # 从第三阶开始计算curr = prev1 + prev2prev1, prev2 = prev2, curr # 滚动更新return prev2
优势:
- 时间复杂度是 O ( n ) O(n) O(n)。
- 空间复杂度降至 O ( 1 ) O(1) O(1),非常高效。
3. 从爬楼梯到斐波那契:算法的深远意义
在解决爬楼梯问题时,我们不仅用到了递归与动态规划,还触碰到了算法的“哲学”内核:
- 化繁为简:将复杂问题分解为小问题,通过递归或迭代求解。
- 优化思维:从朴素递归到记忆化,再到动态规划和空间优化,反映了算法不断进化的过程。
- 广泛应用:斐波那契数列及其变种在动态规划中有着极为广泛的应用,如股票买卖、最长子序列等。
4. 举一反三:实战小题目
爬楼梯问题可以扩展为更多实际场景:
- 不同步数的爬楼梯:每次可以爬1、2或3阶,如何修改代码解决?
- 路径统计:假设楼梯每阶有个性化通行费用,如何找到代价最低的路径?
例如,每次可选择1、3、5阶,代码如下:
def climb_stairs_varied_steps(n, steps):"""通用爬楼梯解法"""dp = [0] * (n + 1)dp[0] = 1 # 到达第0阶的方式只有一种for i in range(1, n + 1):dp[i] = sum(dp[i - step] for step in steps if i - step >= 0)return dp[n]
结语:从爬楼梯到算法巅峰
爬楼梯问题虽小,却涵盖了递归、动态规划、滚动数组等算法的精髓。在算法学习的过程中,我们不仅需要解决问题,更要透过问题看到本质。这道题目告诉我们:从解决问题的简单方法开始,不断优化,就能更接近算法的美和极致。
相关文章:
从爬楼梯到算法世界:动态规划与斐波那契的奇妙邂逅
从爬楼梯到算法世界:动态规划与斐波那契的奇妙邂逅 在算法学习的旅程中,总有一些经典问题让人印象深刻。“爬楼梯问题”就是其中之一,看似简单的题目,却蕴藏了动态规划与斐波那契数列的深刻联系。今天,我就以这个问题…...

centos7使用yum快速安装最新版本Jenkins-2.462.3
Jenkins支持多种安装方式:yum安装、war包安装、Docker安装等。 官方下载地址:https://www.jenkins.io/zh/download 本次实验使用yum方式安装Jenkins LTS长期支持版,版本为 2.462.3。 一、Jenkins基础环境的安装与配置 1.1:基本…...

【vue】【element-plus】 el-date-picker使用cell-class-name进行标记,type=year不生效解决方法
typedete,自定义cell-class-name打标记效果如下: 相关代码: <el-date-pickerv-model"date":clearable"false":editable"false":cell-class-name"cellClassName"type"date"format&quo…...

c++11新特性随笔
1.统一初始化特性 c98中不支持花括号进行初始化,编译时会报错,在11当中初始化可以通过{}括号进行统一初始化。 c98编译报错 c11: #include <iostream> #include <set> #include <string> #include <vector>int main() {std:…...
Linux字符设备驱动开发的详细步骤
1. 确定主设备号 手动指定:明确设备号时,使用register_chrdev_region()静态申请(需确保未被占用)。动态分配:通过alloc_chrdev_region()由内核自动分配主设备号(更灵活,推…...
Nginx 二进制部署与 Docker 部署深度对比
一、核心概念解析 1. 二进制部署 通过包管理器(如 apt/yum)或源码编译安装 Nginx,直接运行在宿主机上。其特点包括: 直接性:与操作系统深度绑定,直接使用系统库和内核功能 。定制化:支持通过编译参数(如 --with-http_ssl_module)启用或禁用模块,满足特定性能需求 。…...

C++23 中 constexpr 的重要改动
文章目录 1. constexpr 函数中使用非字面量变量、标号和 goto (P2242R3)示例代码 2. 允许 constexpr 函数中的常量表达式中使用 static 和 thread_local 变量 (P2647R1)示例代码 3. constexpr 函数的返回类型和形参类型不必为字面类型 (P2448R2)示例代码 4. 不存在满足核心常量…...
CMake ctest
CMake学习–ctest全面介绍 1. 环境准备 确保已安装 cmake 和编译工具: sudo apt update sudo apt install cmake build-essential2. 创建示例项目 假设我们要测试一个简单的数学函数 add(),项目结构如下: math_project/ ├── CMakeList…...

全面解析React内存泄漏:原因、解决方案与最佳实践
在开发React应用时,内存泄漏是一个常见但容易被忽视的问题。如果处理不当,它会导致应用性能下降、卡顿甚至崩溃。由于React的组件化特性,许多开发者可能没有意识到某些操作(如事件监听、异步请求、定时器等)在组件卸载…...
JavaScript学习教程,从入门到精通,Ajax数据交换格式与跨域处理(26)
Ajax数据交换格式与跨域处理 一、Ajax数据交换格式 1. XML (eXtensible Markup Language) XML是一种标记语言,类似于HTML但更加灵活,允许用户自定义标签。 特点: 可扩展性强结构清晰数据与表现分离文件体积相对较大 示例代码࿱…...

【FreeRTOS】事件标志组
文章目录 1 简介1.1事件标志1.2事件组 2事件标志组API2.1创建动态创建静态创建 2.2 删除事件标志组2.3 等待事件标志位2.4 设置事件标志位在任务中在中断中 2.5 清除事件标志位在任务中在中断中 2.6 获取事件组中的事件标志位在任务中在中断中 2.7 函数xEventGroupSync 3 事件标…...

超级扩音器手机版:随时随地,大声说话
在日常生活中,我们常常会遇到手机音量太小的问题,尤其是在嘈杂的环境中,如KTV、派对或户外活动时,手机自带的音量往往难以满足需求。今天,我们要介绍的 超级扩音器手机版,就是这样一款由上海聚告德业文化发…...

【数据可视化-27】全球网络安全威胁数据可视化分析(2015-2024)
🧑 博主简介:曾任某智慧城市类企业算法总监,目前在美国市场的物流公司从事高级算法工程师一职,深耕人工智能领域,精通python数据挖掘、可视化、机器学习等,发表过AI相关的专利并多次在AI类比赛中获奖。CSDN…...

【6G 开发】NV NGC
配置 生成密钥 API Keys 生成您自己的 API 密钥,以便通过 Docker 客户端或通过 NGC CLI 使用 Secrets Manager、NGC Catalog 和 Private Registry 的 NGC 服务 以下个人 API 密钥已成功生成,可供此组织使用。这是唯一一次显示您的密钥。 请妥善保管您的…...
计算机视觉各类任务评价指标详解
文章目录 计算机视觉各类任务评价指标详解一、图像分类(Image Classification)常用指标1. 准确率(Accuracy)2. Top-k Accuracy3. 精确率(Precision)、召回率(Recall)、F1 分数&#…...

SIEMENS PLC程序解读 -Serialize(序列化)SCATTER_BLK(数据分散)
1、程序数据 第12个字节 PI 2、程序数据 第16个字节 PI 3、程序数据 第76个字节 PO 4、程序代码 2、程序解读 图中代码为 PLC 梯形图,主要包含以下指令及功能: Serialize(序列化): 将 SRC_VARIABLEÿ…...

宁德时代25年时代长安动力电池社招入职测评SHL题库Verify测评语言理解数字推理真题
测试分为语言和数字两部分,测试时间各为17分钟,测试正式开始后不能中断或暂停...
python源码打包为可执行的exe文件
文章目录 简单的方式(PyInstaller)特点步骤安装 PyInstaller打包脚本得到.exe文件 简单的方式(PyInstaller) 特点 支持 Python 3.6打包为单文件(–onefile)或文件夹形式自动处理依赖项 步骤 安装 PyIns…...
数据加密技术:从对称加密到量子密码的原理与实战
数据加密技术:从对称加密到量子密码的原理与实战 在网络安全体系中,数据加密是保护信息机密性、完整性的核心技术。从古代的凯撒密码到现代的量子加密,加密技术始终是攻防博弈的关键战场。本文将深入解析对称加密、非对称加密、哈希函数的核…...
高性能的开源网络入侵检测和防御引擎:Suricata介绍
一、Debian下使用Suricata 相较于Windows,Linux环境对Suricata的支持更加完善,操作也更为便捷。 1. 安装 Suricata 在Debian系统上,你可以通过包管理器 apt 轻松安装 Suricata。 更新软件包列表: sudo apt update安装 Suricata: sudo apt …...

【硬核解析:基于Python与SAE J1939-71协议的重型汽车CAN报文解析工具开发实战】
引言:重型汽车CAN总线的数据价值与挑战 随着汽车电子化程度的提升,控制器局域网(CAN总线)已成为重型汽车的核心通信网络。不同控制单元(ECU)通过CAN总线实时交互海量报文数据,这些数据隐藏着车…...
React类组件与React Hooks写法对比
React 类组件 vs Hooks 写法对比 分类类组件(Class Components)函数组件 Hooks组件定义class Component extends React.Componentconst Component () > {}状态管理this.state this.setState()useState()生命周期componentDidMount, componentDidU…...

Uniapp 自定义 Tabbar 实现教程
Uniapp 自定义 Tabbar 实现教程 1. 简介2. 实现步骤2.1 创建自定义 Tabbar 组件2.2 配置 pages.json2.3 在 App.vue 中引入组件 3. 实现过程中的关键点3.1 路由映射3.2 样式设计3.3 图标处理 4. 常见问题及解决方案4.1 页面跳转问题4.2 样式适配问题4.3 性能优化 5. 扩展功能5.…...

记录一次使用面向对象的C语言封装步进电机驱动
简介 (2025/4/21) 本库对目前仅针对TB6600驱动下的42步进电机的基础功能进行了一定的封装, 也是我初次尝试以面向对象的思想去编写嵌入式代码, 和直流电机的驱动步骤相似在调用stepmotor_attach()函数和stepmotor_init()函数之后仅通过结构体数组stepm然后指定枚举变量中的id即…...

Spark-streaming核心编程
1.导入依赖: <dependency> <groupId>org.apache.spark</groupId> <artifactId>spark-streaming-kafka-0-10_2.12</artifactId> <version>3.0.0</version> </dependency> 2.编写代码: 创建Sp…...
Exposure Adjusted Incidence Rate (EAIR) 暴露调整发病率:精准量化疾病风险
1. 核心概念 1.1 传统发病率的局限性 1.1.1 公式与定义 传统发病率公式为新发病例数除以总人口数乘以观察时间。例如在某社区观察1年,有10例新发病例,总人口1000人,发病率即为10/10001=0.01。 此公式假设所有个体暴露时间和风险相同,但实际中个体差异大,如部分人暴露时间…...

vue3+TS+echarts 折线图
需要实现的效果如下 <script setup lang"ts" name"RepsSingleLineChart">import * as echarts from echartsimport { getInitecharts } from /utils/echartimport type { EChartsOption } from echarts// 定义 props 类型interface Props {id: strin…...
MYSQL中为什么不建议delete数据
在 MySQL 中不建议频繁使用 delete 删除数据的原因主要在于性能、数据安全等方面的问题,以下是具体介绍: 性能问题 磁盘空间与碎片:delete 操作只是将数据标记为 “已删除”,并不会立即释放磁盘空间,频繁执行会导致大量…...
Linux多线程技术
什么是线程 在一个程序里的多执行路线就是线程。线程是进程中的最小执行单元,可理解为 “进程内的一条执行流水线”。 进程和线程的区别 进程是资源分配的基本单位,线程是CPU调度的基本单位。 fork创建出一个新的进程,会创建出一个新的拷贝&…...
12个HPC教程汇总!从入门到实战,覆盖分子模拟/材料计算/生物信息分析等多个领域
在科学研究、工程仿真、人工智能和大数据分析等领域,高性能计算 (High Performance Computing, HPC) 正扮演着越来越重要的角色。它通过并行处理、大规模计算资源的整合,极大提升了计算效率,使原本耗时数日的任务能够在数小时内完成。 随着计…...