函数递归
-
递归调用的定义
递归调用是函数嵌套调用的一种特殊形式,函数在调用时,直接或间接调用了自身,就是递归调用。
# 直接调用
def f1():
print('from f1')
f1()
f1()
# 死循环只是为了举例
# 间接调用本身
def f1():
print('from f1')
f2()
def f2():
print('from f2')
f1()
f1()
# 调用函数会产生局部的名称空间,占用内存,因为上述这种调用会无需调用本身,python解释器的内存管理机制为了防止其无限制占用内存,对函数的递归调用做了最大的层级限制 -
递归调用的两个阶段:回溯,递推
回溯
是从外向里一层一层的递归调用下去,回溯阶段必须要有一个明确的结束条件,每进入下一次递归时,问题的规模都因该有所减少(否则,单纯地重复调用自身是没有意义的)
递推
从里向外一层一层结束递归
-
总结递归的使用
-
必须有一个明确的结束条件
-
每次进入更深一层递归时,问题规模相比上次递归都应有所减少
-
递归效率不高,递归层次过多会导致栈溢出(在计算机中,函数调用是通过栈(stack)这种数据结构实现的,每当进入一个函数调用,栈就会加一层栈帧,每当函数返回,栈就会减一层栈帧。由于栈的大小不是无限的,所以,递归调用的次数过多,会导致栈溢出)。
-
-
举例
def age(n):
if n == 1:
return 18
return age(n-1)+2
res = age(5)
print(res)
# 26list_0=[1,[2,[3,[4,[5,[6,[7,[8,[9,]]]]]]]]]
def func(l):
for num in l:
if type(num) is list:
func(num)
else:
print(num,end=' ')
func(list_0)
# 1 2 3 4 5 6 7 8 9
二分法
-
二分法是算法的一种,算法是如何高效地解决问题的思路
nums = [1,2,13,15,23,28,33,57,61,76,85,92,100]
def binary_search(find_num,nums):
print(nums)
if len(nums) == 0:
print('not exists')
return
mid_index = len(nums)//2
if find_num > nums[mid_index]:
nums = nums[mid_index+1:]
binary_search(find_num,nums)
elif find_num < nums[mid_index]:
nums = nums[:mid_index]
binary_search(find_num,nums)
else:
print('Find it')
binary_search(92,nums)
# [1, 2, 13, 15, 23, 28, 33, 57, 61, 76, 85, 92, 100]
# [57, 61, 76, 85, 92, 100]
# [92, 100]
# [92]
# Find it