函数的递归调用

在调用一个函数的过程中直接或间接地调用到了本身。函数的递归调用本质就是一个循环的过程(用函数实现的循环)。

递归调用必须在满足某种条件下结束,不能无限递归调用下去。

1、直接调用自身:

def f1():

print('from f1')

f1()

2、间接调用自身:

def f1():

print('from f1')

f2()

def f2():

print('from f2')

f1()

python不是一门函数式编程语言,无法对递归进行尾递归优化。尾递归优化,即在函数的最后一步(而非最后一行)调用自己。

在Python中,每调用一次函数就会在内存产生一个局部名称空间,所以为了防止递归死循环造成内存溢出,Python对递归的最大深度做了限制,默认为1000层。

# 查看最大深度。

import sys

print(sys.getrecursionlimit())

# 设置最大深度,但仍受限于操作系统栈大小的限制,不推荐修改。

sys.setrecursionlimit()

递归的两个阶段

例如要求出第一个人年龄,这个人比第二个人大10岁,第二个人又比第三个人大10岁,第三个人比第四个人大10岁,第四个人比第五个人大十岁,而第五个人的年龄为18岁,那么我们要求第一个人的年龄,就是:

man1 = man2 + 10

man2 = man3 + 10

man3 = man4 + 10

man4 = man5 + 10

man5 = 10

用代码来写就是这样:

def age(n):

if n == 1:

return 18

return age(n-1) + 10

执行过程:

回溯阶段

不断的向下一层函数递进,这个过程称为回溯。

print(age(5))

# age(5) -> age(4) -> age(3) -> age(2) -> age1(1)

递推阶段

满足结束条件,逐层返回值,这个过程称为递推。

age(5)

58

递归应用

在写代码时,如果出现某个功能需要重复调用自身,这个时候就可以使用到递归。

二分法

对一个从小到大排序的有序数字元素列表,查找某个值在列表中的所在位置。若使用遍历查找,那么时间复杂度为O(n)。我们可以使用二分法查找,每次将值与列表中间的元素比较,若值比中间元素大,则下次从右边查找即可;若值比中间元素小,下次从左边查找即可。每次将值与列表的中间值比较,直到找到该值或无该值。

列表为:

l = [-4,-1,0,1,5,8,9,2,8,12,62,23,63]

Python实现二分法:

def dichotomy(n,num_list):

# 对列表排序。

num_list.sort()

if len(num_list) == 0:

# 表示列表为空,无该值.

return

median = len(num_list) // 2

if n > num_list[median]:

# 如果值大于中间值。

new_list = num_list[median+1:]

dichotomy(n,new_list)

elif n < num_list[median]:

# 如果值小于中间值。

new_list = num_list[:median]

dichotomy(n,new_list)

else:

# 不大不小表示找到该值。

print(f'{n} index is {median}')

阶乘

一个正整数的阶乘(factorial)是所有小于及等于该数的正整数的积,并且0的阶乘为1。自然数n的阶乘写作n!。

任何大于等于1 的自然数n 阶乘表示方法:

n! = n * n(n-1)!

Python代码实现:

def factorial(n):
if n == 1:
return n
return factorial(n-1) * n