如果一個函數(shù)直接或者間接調(diào)用了自己,那么就形成了遞歸(recursion),比如斐波那契數(shù)列的一個實現(xiàn)
def fib(n): if n <= 2: return 1 else: return fib(n - 1) + fib(n - 2)
遞歸一定要有結(jié)束條件,否則就形成了死循環(huán), 比如下面的例子:
def a(): b()def b(): a() if __name__ == '__main__': a()
很快 就會拋出一個異常:RuntimeError: maximum recursion depth exceeded
會什么報這個異常,很簡單,我們都知道子程序調(diào)用(call)需要壓棧出棧,如果無限遞歸調(diào)用,那么就一直壓棧,沒有出棧,內(nèi)存消耗也越來愈多。python比較高級,直接拋出這個異常,結(jié)束程序運行。
前面的文章提到協(xié)程(coroutine)這個概念,不管是generator還是greenlet,語法看起來都很像函數(shù)調(diào)用,但事實上并不是,協(xié)程只是在切換的時候把當前調(diào)用棧中的信息存儲了起來:
“all local state is retained, including the current bindings of local variables, the instruction pointer, and the internal evaluation stack. When the execution is resumed by calling one of the generator’s methods, the function can proceed exactly a