递归和生成器函数

递归

如果函数包含了对其自身的调用,该函数就是递归。递归广泛应用于语言识别和使用递归函数的数学应用中。例如:斐波那契数列和求阶乘等。下面就上面两种使用举例:
斐波那契数列:

In [12]: def fib(n):
                if n==0:
                    return 1
                if n==1:
                    return 1
                return fib(n-2)+fib(n-1)

In [18]: fib(10)
Out[18]: 89

阶乘:

In [19]: def fn(n):
   ....:     if n==0:
   ....:         return 1
   ....:     if n==1:
   ....:         return 1
   ....:     return n*fn(n-1)

In [20]: fn(10)
Out[20]: 3628800

由于递归总是涉及到压栈和出栈,所以递归函数在Python中非常慢,并且有深度限制,所以尽量避免使用递归。在Python中递归的限制可以通过 sys.getrecursionlimit() 得到深度限制,通过sys.setrecursionlimit调整递归深度限制。
注意:循环都可以转化为递归,但不是所有递归都可以转化为循环。

生成器函数

生成器函数是一个带yield语句的函数。一个函数或者子程序只能返回一次,但一个生成器能暂停执行并返回一个中间的结果,这就是yield的功能,返回一个值给调用者并暂停执行。当生成器被next()方法调用的时候,它会准确的从离开地方继续。
请注意yield和return的区别:

  • yield 弹出值,暂停函数

  • return返回值, 并且结束函数

  • 当yield和return同时存在时, return的返回值会被忽略,但是return依然可以中断生成器

In [50]: def gen(x):
   ....:     for i in range(10):
   ....:         if i ==3:
   ....:             return i
   ....:         yield i
   
In [51]: g = gen(10)

In [52]: for i in g:
   ....:     print(i) 
0
1
2

注意:可以使用生成器来替换递归,同时所有的递归,都可以用生成器替换。

In [54]: def fn1():
                ret =1
                index=1
                while True:
                    yield ret
                    index += 1
                    ret *= index

In [55]: g1=fn1()

In [56]: next(g1)
Out[56]: 1

In [57]: next(g1)
Out[57]: 2

In [58]: next(g1)
Out[58]: 6

In [59]: next(g1)
Out[59]: 24

In [60]: next(g1)
Out[60]: 120
def g(n):
    def factorial():
        ret = 1
        idx = 1
        while True:
            yield ret
            idx += 1
            ret *= idx
    gen = factorial()
    for _ in range(n-1):
        next(gen)
    return next(gen)

In [63]: fn2(5)
Out[63]: 120

In [64]: fn2(6)
Out[64]: 720

生成器函数的列表传参
在Python2 中使用:

In [1]: lst = [1,2,3,4,5,7,9]

In [2]: def gen(lst):
   ...:     for x in lst:
   ...:         yield x        

In [3]: g=gen(lst)

In [4]: for i in g:
   ...:     print(i) 
1
2
3
4
5
7
9

在Python3 中使用:

In [5]: def gen1(lst):
   ...:     yield from lst

In [6]: g1=gen1(lst)

In [7]: for j in g1:
   ...:     print(j)
1
2
3
4
5
7
9
posted @ 2016-10-01 17:27  ProfiBus  阅读(243)  评论(0编辑  收藏  举报