python学习(八)之递归函数
2023-09-11 14:20:36 时间
递归函数的优点是定义简单,逻辑清晰。理论上,所有的递归函数都可以写成循环的方式,但循环的逻辑不如递归清晰。
使用递归函数需要注意防止栈溢出。在计算机中,函数调用是通过栈(stack)这种数据结构实现的,每当进入一个函数调用,栈就会加一层栈帧,每当函数返回,栈就会减一层栈帧。由于栈的大小不是无限的,所以,递归调用的次数过多,会导致栈溢出。可以试试fact(1000):
>>> fact(1000)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "<stdin>", line 4, in fact
File "<stdin>", line 4, in fact
RuntimeError: maximum recursion depth exceeded in comparison
解决递归调用栈溢出的方法是通过尾递归优化,事实上尾递归和循环的效果是一样的,所以,把循环看成是一种特殊的尾递归函数也是可以的。
def fact(n):
return fact_iter(n, 1)
def fact_iter(num, product):
if num == 1:
return product
return fact_iter(num - 1, num * product)
在函数返回的时候,调用自身本身,并且,return语句不能包含表达式。这样,编译器或者解释器就可以把尾递归做优化,使递归本身无论调用多少次,都只占用一个栈帧,不会出现栈溢出的情况。
上面的fact(n)函数由于return n * fact(n - 1)引入了乘法表达式,所以就不是尾递归了。要改成尾递归方式,需要多一点代码,主要是要把每一步的乘积传入到递归函数中:
尾递归调用时,如果做了优化,栈不会增长,因此,无论多少次调用也不会导致栈溢出。
遗憾的是,大多数编程语言没有针对尾递归做优化,Python解释器也没有做优化,所以,即使把上面的fact(n)函数改成尾递归方式,也会导致栈溢出.
Hanoi
# -*- coding: utf-8 -*-
def f(n, a, b, c):
if n == 1:
print(a, '-->', c);
else:
f(n-1, a, c, b);
f(1, a, b, c);
f(n-1, b, a, c);
f(3, 'A', 'B', 'C');
相关文章
- 人工智能时代,应立即学习python
- 卧槽,又来一个 Python 神器!!
- 24岁非计算机专业工科妹子裸辞转行Python程序员,自学Python三个月,零基础如何系统学习python,从入门到精通?
- 如何快速学习python,学好python?能通过Python赚到的第一笔钱,有哪些经验可以分享吗?
- 36 python - 文件打开关闭
- 学习python web 需要的掌握技能、框架、必备工具、
- Python爬虫技术--基础篇--图形界面
- 《Python密码学编程》——1.11 双重强度加密
- 《Python Cookbook(第3版)中文版》——1.13 通过公共键对字典列表排序
- 【21天学习经典算法】折半查找与折半插入排序(附Python完整代码)
- 零基础怎么转行IT ?为什么要学习python?(下)
- (数据科学学习手札31)基于Python的网络数据采集(初级篇)
- (数据科学学习手札30)朴素贝叶斯分类器的原理详解&Python与R实现
- (数据科学学习手札14)Mean-Shift聚类法简单介绍及Python实现
- 学习opencv: 获取图像最大连通域 c++和python版
- Python学习---JSONP学习180130
- Python学习---函数的学习1209[all]
- Python学习-将list列表写入文件并读取方法汇总
- 学习笔记(33):Python网络编程&并发编程-进程池线程池
- python 连接sqlserver
- python面试题--判断一个字符串中的括号“(”“)”是否成对出现