Python递归函数深度解析:从原理到实战
Python递归函数深度解析:从原理到实战
递归是计算机科学中重要的编程范式,也是算法设计的核心思想之一。本文将通过20+实战案例,带你深入理解Python递归函数的精髓,掌握递归算法的实现技巧。
一、递归函数核心原理
1.1 递归三要素
- 基线条件:递归终止的条件
- 递归条件:问题分解的规则
- 状态传递:参数的状态变化
简单点说就是:自己调用自己,必须要有出口
1.2 执行过程解析
def countdown(n):if n <= 0: # 基线条件print("Lift off!")else: # 递归条件print(n)countdown(n-1) # 状态传递countdown(3)
"""
输出:
3
2
1
Lift off!
"""
二、基础递归模式
2.1 数值计算
阶乘计算
def factorial(n):return 1 if n == 1 else n * factorial(n-1)print(factorial(5)) # 120
斐波那契数列
def fib(n):return n if n <= 1 else fib(n-1) + fib(n-2)print([fib(i) for i in range(10)])
# [0, 1, 1, 2, 3, 5, 8, 13, 21, 34]
2.2 字符串处理
反转字符串
def reverse_str(s):return s if len(s) <= 1 else reverse_str(s[1:]) + s[0]print(reverse_str("hello")) # olleh
回文判断
def is_palindrome(s):if len(s) < 2:return Trueif s[0] != s[-1]:return Falsereturn is_palindrome(s[1:-1])print(is_palindrome("madam")) # True
三、数据结构处理
3.1 列表深度处理
def deep_sum(arr):total = 0for item in arr:if isinstance(item, list):total += deep_sum(item)else:total += itemreturn totalnested_list = [1, [2, [3, 4], 5], 6]
print(deep_sum(nested_list)) # 21
3.2 字典树遍历
def traverse_tree(node, level=0):print(' '*level + node['name'])for child in node.get('children', []):traverse_tree(child, level+1)tree = {'name': 'Root','children': [{'name': 'Child1'},{'name': 'Child2', 'children': [{'name': 'Grandchild'}]}]
}traverse_tree(tree)
"""
输出:
RootChild1Child2Grandchild
"""
四、经典算法实现
4.1 汉诺塔问题
def hanoi(n, source, target, auxiliary):if n > 0:hanoi(n-1, source, auxiliary, target)print(f"移动圆盘 {n} 从 {source} 到 {target}")hanoi(n-1, auxiliary, target, source)hanoi(3, 'A', 'C', 'B')
"""
输出:
移动圆盘 1 从 A 到 C
移动圆盘 2 从 A 到 B
移动圆盘 1 从 C 到 B
移动圆盘 3 从 A 到 C
移动圆盘 1 从 B 到 A
移动圆盘 2 从 B 到 C
移动圆盘 1 从 A 到 C
"""
4.2 快速排序
def quicksort(arr):if len(arr) <= 1:return arrpivot = arr[len(arr)//2]left = [x for x in arr if x < pivot]middle = [x for x in arr if x == pivot]right = [x for x in arr if x > pivot]return quicksort(left) + middle + quicksort(right)print(quicksort([3,6,8,10,1,2,1])) # [1, 1, 2, 3, 6, 8, 10]
五、高级递归技巧
5.1 记忆化优化
from functools import lru_cache@lru_cache(maxsize=None)
def fibonacci(n):return n if n <= 1 else fibonacci(n-1) + fibonacci(n-2)print(fibonacci(50)) # 12586269025(无优化时无法计算)
5.2 尾递归优化
def factorial(n, acc=1):return acc if n == 0 else factorial(n-1, acc*n)print(factorial(5)) # 120
六、递归调试技巧
6.1 调用栈可视化
def factorial_debug(n, depth=0):print(f"{' '*depth}-> factorial({n})")if n == 1:result = 1else:result = n * factorial_debug(n-1, depth+1)print(f"{' '*depth}<- {result}")return resultfactorial_debug(3)
"""
输出:
-> factorial(3)-> factorial(2)-> factorial(1)<- 1<- 2
<- 6
"""
七、递归的替代方案
7.1 迭代实现
def factorial_iter(n):result = 1for i in range(1, n+1):result *= ireturn resultprint(factorial_iter(5)) # 120
7.2 生成器实现
def traverse_tree_iter(node):stack = [(node, 0)]while stack:node, level = stack.pop()yield (node['name'], level)for child in reversed(node.get('children', [])):stack.append((child, level+1))for name, level in traverse_tree_iter(tree):print(' '*level + name)
八、最佳实践与注意事项
8.1 适用场景
- 树形结构处理
- 分治算法
- 数学递推公式
- 回溯算法
8.2 风险规避
- 栈溢出风险(Python默认递归深度约1000层)
- 重复计算问题
- 时间复杂度失控
- 内存占用过高
8.3 性能优化方案
- 使用记忆化缓存(@lru_cache)
- 转换为迭代实现
- 尾递归优化(Python原生不支持,需自行实现)
- 限制递归深度
import sysdef safe_recursion(func):def wrapper(*args):wrapper.depth += 1if wrapper.depth > sys.getrecursionlimit() - 100:raise RecursionError("超出安全递归深度")try:return func(*args)finally:wrapper.depth -= 1wrapper.depth = 0return wrapper@safe_recursion
def deep_recursion(n):return 1 if n == 0 else deep_recursion(n-1) + 1
九、递归思维训练
9.1 路径搜索问题
def find_path(matrix, start, end, path=[]):path = path + [start]if start == end:return [path]if matrix[start[0]][start[1]] == 0:return []paths = []for direction in [(0,1), (1,0), (0,-1), (-1,0)]:next_pos = (start[0]+direction[0], start[1]+direction[1])if 0 <= next_pos[0] < len(matrix) and \0 <= next_pos[1] < len(matrix[0]) and \next_pos not in path:newpaths = find_path(matrix, next_pos, end, path)paths += newpathsreturn pathsmaze = [[1,1,0,1],[1,1,1,0],[0,1,1,1]
]
print(find_path(maze, (0,0), (2,3)))
十、总结提升
掌握递归不仅是学习编程技巧,更是培养抽象思维和问题分解能力的重要途径。合理运用递归,可以让复杂问题迎刃而解!
相关文章:
Python递归函数深度解析:从原理到实战
Python递归函数深度解析:从原理到实战 递归是计算机科学中重要的编程范式,也是算法设计的核心思想之一。本文将通过20实战案例,带你深入理解Python递归函数的精髓,掌握递归算法的实现技巧。 一、递归函数核心原理 1.1 递归三要…...
OpenGL学习笔记(五):Textures 纹理
文章目录 纹理坐标纹理环绕方式纹理过滤——处理纹理分辨率低的情况多级渐远纹理Mipmap——处理纹理分辨率高的情况加载与创建纹理 ( <stb_image.h> )生成纹理应用纹理纹理单元练习1练习2练习3练习4 通过上一篇着色部分的学习,我们可以…...
【TypeScript】基础:数据类型
文章目录 TypeScript一、简介二、类型声明三、数据类型anyunknownnervervoidobjecttupleenumType一些特殊情况 TypeScript 是JavaScript的超集,代码量比JavaScript复杂、繁多;但是结构更清晰 一、简介 为什么需要TypeScript? JavaScript的…...
Notepad++消除生成bak文件
设置(T) ⇒ 首选项... ⇒ 备份 ⇒ 勾选 "禁用" 勾选禁用 就不会再生成bak文件了 notepad怎么修改字符集编码格式为gbk 如图所示...
Android NDK
Android NDK环境 D:\Android SDK\ndk\25.2.9519653 使用clang而不用gcc D:\Android SDK\ndk\25.1.8937393\toolchains\llvm\prebuilt\windows-x86_64\bin\clang --version 查看是否安装成功clang ptrace 在 C 语言中,ptrace 已经被 Linux 内核实现࿰…...
内部知识库助力组织智力激发与信息共享实现业绩增长
内容概要 内部知识库是企业知识管理的核心组件,具有不可估量的重要性。通过构建有效的知识库,组织能够将孤立的知识和信息整合成为一个系统性的体,极大提高员工访问和利用这些信息的能力。这不仅简化了决策过程,还通过减少重复劳…...
通过F12收集的信息
按 F12 键打开浏览器的开发者工具(DevTools)可以获取部分操作系统和中间件信息,但能力有限。以下是具体说明: 一、通过 F12 收集的信息 1. 客户端操作系统信息 - Console 控制台 通过 JavaScript 直接获取客户端操作系统信息&am…...
用Python替代OpenMV IDE显示openmv USB 图像
原理是利用openmv的usb模仿串口,然后用Python代码打开串口接收 能替代openmv ide 跑48帧图像 Python端需要的依赖: 需要的是: from ultralytics import YOLO import cv2 import numpy as np from serial import Serial import time from co…...
c语言:编译和链接(详解)
前言 要将编译和链接,就不得不提及编译器是如何运作的,虽然这部分知识是针对于要创造编译器和创作语言的人所需要清楚的,但作为c语言的学习者也需要了解一下,修炼内功,尤其是对于想学习c的人而言。 编译器的运作过程…...
数据结构【单链表操作大全详解】【c语言版】(只有输入输出为了方便用的c++)
单链表操作的C/C实现详解 在数据结构中,单链表是一种非常基础且重要的数据结构。它由一系列节点组成,每个节点包含数据和指向下一个节点的指针。今天我们就来深入探讨用C/C实现的单链表及其各种操作。 一、单链表的定义 const int N 1e5; //单链表 t…...
leetcode27.删除有序数组中的重复项
目录 问题描述判题标准示例提示 具体思路思路一思路二 代码实现 问题描述 给你一个非严格递增排列的数组nums,请你原地删除重复出现的元素,使每个元素只出现一次,返回删除后数组的新长度。元素的相对顺序应该保持一致 。然后返回nums中唯一元…...
[c语言日寄]越界访问:意外的死循环
【作者主页】siy2333 【专栏介绍】⌈c语言日寄⌋:这是一个专注于C语言刷题的专栏,精选题目,搭配详细题解、拓展算法。从基础语法到复杂算法,题目涉及的知识点全面覆盖,助力你系统提升。无论你是初学者,还是…...
【c++11】包装器
🔥个人主页:Quitecoder 🔥专栏:c笔记仓 包装器(Wrapper) 是一个常见的编程设计模式,通常用于封装或“包装”某个现有的对象、函数、数据结构或者操作,以提供额外的功能或简化接口。…...
信息学奥赛一本通 1422:【例题1】活动安排
【题目链接】 ybt 1422:【例题1】活动安排 【题目考点】 1. 贪心 【解题思路】 该题属于区间选点问题,ybt 1324:【例6.6】整数区间 是给定一些区间,选择一些点使得每个区间范围内至少有1个点。 本题为:给定一些区…...
数据库、数据仓库、数据湖有什么不同
数据库、数据仓库和数据湖是三种不同的数据存储和管理技术,它们在用途、设计目标、数据处理方式以及适用场景上存在显著差异。以下将从多个角度详细说明它们之间的区别: 1. 数据结构与存储方式 数据库: 数据库主要用于存储结构化的数据&…...
llama.cpp LLM_CHAT_TEMPLATE_DEEPSEEK_3
llama.cpp LLM_CHAT_TEMPLATE_DEEPSEEK_3 1. LLAMA_VOCAB_PRE_TYPE_DEEPSEEK3_LLM2. static const std::map<std::string, llm_chat_template> LLM_CHAT_TEMPLATES3. LLM_CHAT_TEMPLATE_DEEPSEEK_3References 不宜吹捧中国大语言模型的同时,又去贬低美国大语言…...
深度学习的应用场景及常用技术
深度学习作为机器学习的一个重要分支,在众多领域都有广泛的应用,以下是一些主要的应用场景及常用技术。 1.应用场景 1. 计算机视觉 图像分类 描述:对图像中的内容进行分类,识别出图像中物体所属的类别。例如,在安防领…...
小程序项目-购物-首页与准备
前言 这一节讲一个购物项目 1. 项目介绍与项目文档 我们这里可以打开一个网址 https://applet-base-api-t.itheima.net/docs-uni-shop/index.htm 就可以查看对应的文档 2. 配置uni-app的开发环境 可以先打开这个的官网 https://uniapp.dcloud.net.cn/ 使用这个就可以发布到…...
网件r7000刷回原厂固件合集测评
《网件R7000路由器刷回原厂固件详解》 网件R7000是一款备受赞誉的高性能无线路由器,其强大的性能和可定制性吸引了许多高级用户。然而,有时候用户可能会尝试第三方固件以提升功能或优化网络性能,但这也可能导致一些问题,如系统不…...
微信登录模块封装
文章目录 1.资质申请2.combinations-wx-login-starter1.目录结构2.pom.xml 引入okhttp依赖3.WxLoginProperties.java 属性配置4.WxLoginUtil.java 后端通过 code 获取 access_token的工具类5.WxLoginAutoConfiguration.java 自动配置类6.spring.factories 激活自动配置类 3.com…...
MongoDB学习和应用(高效的非关系型数据库)
一丶 MongoDB简介 对于社交类软件的功能,我们需要对它的功能特点进行分析: 数据量会随着用户数增大而增大读多写少价值较低非好友看不到其动态信息地理位置的查询… 针对以上特点进行分析各大存储工具: mysql:关系型数据库&am…...
【入坑系列】TiDB 强制索引在不同库下不生效问题
文章目录 背景SQL 优化情况线上SQL运行情况分析怀疑1:执行计划绑定问题?尝试:SHOW WARNINGS 查看警告探索 TiDB 的 USE_INDEX 写法Hint 不生效问题排查解决参考背景 项目中使用 TiDB 数据库,并对 SQL 进行优化了,添加了强制索引。 UAT 环境已经生效,但 PROD 环境强制索…...
最新SpringBoot+SpringCloud+Nacos微服务框架分享
文章目录 前言一、服务规划二、架构核心1.cloud的pom2.gateway的异常handler3.gateway的filter4、admin的pom5、admin的登录核心 三、code-helper分享总结 前言 最近有个活蛮赶的,根据Excel列的需求预估的工时直接打骨折,不要问我为什么,主要…...
uniapp微信小程序视频实时流+pc端预览方案
方案类型技术实现是否免费优点缺点适用场景延迟范围开发复杂度WebSocket图片帧定时拍照Base64传输✅ 完全免费无需服务器 纯前端实现高延迟高流量 帧率极低个人demo测试 超低频监控500ms-2s⭐⭐RTMP推流TRTC/即构SDK推流❌ 付费方案 (部分有免费额度&#x…...
Web 架构之 CDN 加速原理与落地实践
文章目录 一、思维导图二、正文内容(一)CDN 基础概念1. 定义2. 组成部分 (二)CDN 加速原理1. 请求路由2. 内容缓存3. 内容更新 (三)CDN 落地实践1. 选择 CDN 服务商2. 配置 CDN3. 集成到 Web 架构 …...
关键领域软件测试的突围之路:如何破解安全与效率的平衡难题
在数字化浪潮席卷全球的今天,软件系统已成为国家关键领域的核心战斗力。不同于普通商业软件,这些承载着国家安全使命的软件系统面临着前所未有的质量挑战——如何在确保绝对安全的前提下,实现高效测试与快速迭代?这一命题正考验着…...
Angular微前端架构:Module Federation + ngx-build-plus (Webpack)
以下是一个完整的 Angular 微前端示例,其中使用的是 Module Federation 和 npx-build-plus 实现了主应用(Shell)与子应用(Remote)的集成。 🛠️ 项目结构 angular-mf/ ├── shell-app/ # 主应用&…...
wpf在image控件上快速显示内存图像
wpf在image控件上快速显示内存图像https://www.cnblogs.com/haodafeng/p/10431387.html 如果你在寻找能够快速在image控件刷新大图像(比如分辨率3000*3000的图像)的办法,尤其是想把内存中的裸数据(只有图像的数据,不包…...
C++实现分布式网络通信框架RPC(2)——rpc发布端
有了上篇文章的项目的基本知识的了解,现在我们就开始构建项目。 目录 一、构建工程目录 二、本地服务发布成RPC服务 2.1理解RPC发布 2.2实现 三、Mprpc框架的基础类设计 3.1框架的初始化类 MprpcApplication 代码实现 3.2读取配置文件类 MprpcConfig 代码实现…...
使用SSE解决获取状态不一致问题
使用SSE解决获取状态不一致问题 1. 问题描述2. SSE介绍2.1 SSE 的工作原理2.2 SSE 的事件格式规范2.3 SSE与其他技术对比2.4 SSE 的优缺点 3. 实战代码 1. 问题描述 目前做的一个功能是上传多个文件,这个上传文件是整体功能的一部分,文件在上传的过程中…...
