Python装饰器结合递归原理解析

作者:JonnyJiang-zh 时间:2023-07-13 22:24:00 

代码如下:


import functools

def memoize(fn):
 print('start memoize')
 known = dict()

@functools.wraps(fn)
 def memoizer(*args):
   if args not in known:
     print('memorize %s'%args)
     # known[args] = fn(*args)
   for k in known.keys():
       print('%s : %s'%(k, known[k]), end = ' ')
   print()
   # return known[args]
 return memoizer

@memoize
def nsum(n):
 print('now is %s'%n)
 assert (n >= 0), 'n must be >= 0'
 return 0 if n == 0 else n + nsum(n - 1)

@memoize
def fibonacci(n):
 assert (n >= 0), 'n must be >= 0'
 return n if n in (0, 1) else fibonacci(n - 1) + fibonacci(n - 2)

if __name__ == '__main__':
 print(nsum(10))
 print(fibonacci(10))

输出如下:

start memoize
start memoize
memorize 10

None
memorize 10

None

对比代码(把注释的地方去掉后)的输出:


start memoize
start memoize
memorize 10
now is 10
memorize 9
now is 9
memorize 8
now is 8
memorize 7
now is 7
memorize 6
now is 6
memorize 5
now is 5
memorize 4
now is 4
memorize 3
now is 3
memorize 2
now is 2
memorize 1
now is 1
memorize 0
now is 0
(0,) : 0
(0,) : 0 (1,) : 1
(0,) : 0 (1,) : 1 (2,) : 3
(0,) : 0 (1,) : 1 (2,) : 3 (3,) : 6
(0,) : 0 (1,) : 1 (2,) : 3 (3,) : 6 (4,) : 10
(0,) : 0 (1,) : 1 (2,) : 3 (3,) : 6 (4,) : 10 (5,) : 15
(0,) : 0 (1,) : 1 (2,) : 3 (3,) : 6 (4,) : 10 (5,) : 15 (6,) : 21
(0,) : 0 (1,) : 1 (2,) : 3 (3,) : 6 (4,) : 10 (5,) : 15 (6,) : 21 (7,) : 28
(0,) : 0 (1,) : 1 (2,) : 3 (3,) : 6 (4,) : 10 (5,) : 15 (6,) : 21 (7,) : 28 (8,) : 36
(0,) : 0 (1,) : 1 (2,) : 3 (3,) : 6 (4,) : 10 (5,) : 15 (6,) : 21 (7,) : 28 (8,) : 36 (9,) : 45
(0,) : 0 (1,) : 1 (2,) : 3 (3,) : 6 (4,) : 10 (5,) : 15 (6,) : 21 (7,) : 28 (8,) : 36 (9,) : 45 (10,) : 55

通过取消注释的对比,可以得到如下结论:

  • 装饰器memoize实际上对于函数nsum()只执行了第一次加载的时候的预处理,然后就是nsum = memoizer。

  • 装饰器的实质是通过functools.wraps(fn)获得函数的名字,便于nsum.__name__ ==nsum,并将参数传至memoize(*args),也就是*args。

  • 装饰器通过memory(),和外面的装饰器获得的函数,在内部对函数进行功能改造。在上例子中,通过known[args] = fn(*args)先执行fn函数,即上例子中nsum(10),然后就进入递归,t同时调用memoizer()和nsum()函数10次,且先memoizer再nsum,而且每次都在``known[args] = fn(*args)`进入递归,也就是每次nsum的执行,故,对于为什么打印konwn中的元素是集中在一起的解释就知道了,到了n == 0,才跳出递归,故,known的第一个元素是0,然后就循环往复。

  • 最后,其实,递归函数执行的是fn(*args),即nsum()。

来源:https://www.cnblogs.com/JonnyJiang-zh/p/13216770.html

标签:python,装饰器,结合,递归
0
投稿

猜你喜欢

  • 关于useSSL=false和true的区别及说明

    2024-01-25 03:27:40
  • javascript实现checkbox全选的代码

    2024-04-16 10:38:11
  • Oracle SQL性能优化系列学习一

    2010-07-26 13:14:00
  • JS实现普通轮播图特效

    2024-04-17 10:19:52
  • python机器学习实现神经网络示例解析

    2022-10-24 03:45:09
  • MySQL的Query Cache原理分析

    2024-01-25 05:20:29
  • 基于Python+Appium实现京东双十一自动领金币功能

    2022-10-15 19:21:04
  • Python字典对象实现原理详解

    2023-06-11 01:53:21
  • MySQL查询重写插件的使用

    2024-01-27 15:55:58
  • thinkPHP利用ajax异步上传图片并显示、删除的示例

    2024-06-07 15:34:26
  • python对输出的奇数偶数排序实例代码

    2023-02-17 17:55:16
  • php5.4传引用时报错问题分析

    2023-11-14 15:03:17
  • python+matplotlib实现动态绘制图片实例代码(交互式绘图)

    2022-06-16 15:43:08
  • js同时按下两个方向键

    2024-05-02 17:21:45
  • Python3爬虫里关于Splash负载均衡配置详解

    2022-11-24 22:54:19
  • 基于Golang实现内存数据库的示例详解

    2024-01-20 05:23:18
  • Javascript 利用 DOM 特性的两个小技巧

    2009-02-28 14:07:00
  • python多线程方法详解

    2023-10-16 02:46:31
  • mysql命令行中执行sql的几种方式总结

    2024-01-14 16:11:33
  • 使用Python Fast API发布API服务的过程详解

    2022-05-19 20:03:44
  • asp之家 网络编程 m.aspxhome.com