Python攻克之路-递归函数
递归函数
在数理逻辑和计算机科学中,递归函数或μ-递归函数是一类从自然数到自然数的函数。直觉上递归函数是"可计算的"。事实上在可计算性理论中已经证明了它确实是图灵机的可计算函数。递归函数与原始递归函数相关,而且递归函数的归纳定义(见下)创建在原始递归函数之上。但不是所有递归函数都是原始递归函数——其中最著名的是阿克曼函数。
其他等价的函数类是λ-递归函数和马尔可夫算法可计算的函数。
所有递归函数的集合叫做R。
通俗描述:在函数内部,可以调用其他函数, 如果一个函数在内部调用自身,这个函数就是递归函数
实现阶乘例子1:
5!=5*4*3*2*1=120 7!=7*6*5*3*2*1=5040
函数需求:f(5)=120 f(7)=5040
实现思路:n*(n-1)*(n-2)..*1,这是一个所需要结果,但是存在一个问题,无法确定传入的数,如7就7次,8就8次,可以通过for循环实现,做一个多次循环的结果,range(1,n)传入相乘次数,再做一个累次的相乘,使用一个变量来累乘,ret=ret*i,先赋值ret=1,ret=1*1..ret=1*n做成一个循环,再使用return返回结果,还有一个问题range(1,n)是从1开始取值,但是是取不到n,所以要修改成n+1
In [14]: def f(n): ...: ret=1 ...: for i in range(1,n+1): ...: ret=ret*i ...: return ret In [15]: f(5) Out[15]: 120 In [16]: f(7) Out[16]: 5040
递归实现阶乘
描述:在函数可以引用其他函数,可以执行自身的函数,阶乘最直观是n*(n-1),n*recursion(n-1)原理如下图,而且有个问题,
这个递归函数会一直向下走,需要有结束条件
递归特点:
a.引用自身函数
b.有一个结束条件
c.凡是可以使用递归写的,循环都可以实现,而且在一定情况下,递归的效率是比较低(取决于使用场景)
注:在计算机中,函数调用是能过栈(stack)这种数据结构实现的,每当进入一个函数调用,栈就会加一层栈帧,每当函数返回加,栈就会减少一层栈帧.由于栈的大小不是无限的,所以,递归调用的次数过多,会导致栈溢出.
In [17]: def recursion(n): ...: if n==1: ##结束条件 ...: return 1 ##当n=1时,应该停,5*4*3*2*recursion(1),正常情况下在这里需要函数返回1,就可以 ...: return n*recursion(n-1) ##引用自身,会产生无限的调用 In [18]: recursion(5) Out[18]: 120
实例2(斐波那契数列)
斐波那契数列(Fibonacci sequence),又称黄金分割数列、因数学家列昂纳多·斐波那契(Leonardoda Fibonacci)以兔子繁殖为例子而引入,故又称为“兔子数列”,指的是这样一个数列:1、1、2、3、5、8、13、21、34、……在数学上,斐波纳契数列以如下被以递归的方法定义:F(0)=0,F(1)=1, F(n)=F(n-1)+F(n-2)(n>=2,n∈N*)在现代物理、准晶体结构、化学等领域,斐波纳契数列都有直接的应用
简单的理解:规律是后面一个数是前面两个数字的相加
#数列为:0、1、1、2、3、5、8、13、21、34、……
#使用函数实现效果:f(3)=1 f(8)=13,需要使用两个变量来实现,实现f(8)=f(7)+f(6)
描述:fibo函数需要传入一个参数,可以直接return,return fibo(n-1)+fibo(n-2),再通过这个函数去调用,然后需要一个结束条件,如fibo(3)=fibo(2)+fibo(1)=fibo(1)+fibo(0)+fibo(1),其中的fibo(2)和fibo(1)还要再次调用,数列最开始是0、1,
In [19]: def fibo(n): ...: if n <= 1: #当n=1时,返回1,当n=0时,返回0,不会再走一次下面 ...: return n ...: return fibo(n-1)+fibo(n-2) In [20]: fibo(8) ##默认从第二个数是1时开始,当n等于第8个数时 Out[20]: 21 #另一种写法: In [21]: def fibo(n): ...: if n==0 or n==1: ...: return n ...: return fibo(n-1)+fibo(n-2) In [22]: fibo(8) Out[22]: 21