文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Python装饰器原理——偷梁换柱

2023-01-31 00:47

关注

本篇博客写于学完  金角大王Alex和海峰老师  于  老男孩2016年周末班S14期第4周03~08章节   装饰器  的相关内容,为对该部分视频的一个思考与补充。

下面举例子,代码如下:

例1,不使用语法糖(@xxx):

 1 import time
 2 
 3 def timer(func):
 4     def deco():
 5         start_time = time.time()
 6         func()
 7         end_time = time.time()
 8         print('running time:',end_time-start_time)
 9         print('func的内存地址:',func)
10         print('deco的内存地址:',deco)
11         print('最后test的内存地址:',test)
12     return deco
13 
14 def test():
15     time.sleep(1)
16     print('in the test.')
17 
18 print('原本test的内存地址:',test)
19 test = timer(test)
20 test()  #调用test,test和deco指向内存中的同一个地址,这时是在调用deco

声明:1.被装饰的函数为test(),其功能为睡1s并输出'in the test.'。

           2装饰器函数为timer(),其功能是使test()函数能额外输出其运行时间。

输出结果如下:

原本test的内存地址: <function test at 0x0000000003083268>
in the test.
running time: 5.694325685501099
func的内存地址: <function test at 0x0000000003083268>
deco的内存地址: <function timer.<locals>.deco at 0x0000000003C84048>
最后test的内存地址: <function timer.<locals>.deco at 0x0000000003C84048>

 

让我们在调试模式下看看每个变量名的内存地址吧:

 

由图片可知,首次定义test函数时,test函数的函数体在内存中的地址为:<function test at 0x0000000003083268>,即test=<function test at 0x0000000003083268>

其后,第19行的代码test = timer(test)调用timer函数,将test的内存地址赋给func,此时func=test=<function test at 0x0000000003083268>

调用timer函数,deco函数的函数体放于内存中的<function timer.<locals>.deco at 0x0000000003C84048>,即demo=<function timer.<locals>.deco at 0x0000000003C84048>

19 test = timer(test)   调用timer函数,timer函数最后  12 return deco返回deco给test,此时test=demo=<function timer.<locals>.deco at 0x0000000003C84048>,即函数名(变量)

test与deco指向同一处内存地址,test函数已经不指向原来的内存地址了,test函数已经不是原来的test函数,偷梁换柱完成。

为了有更好的理解,可以自己调试看看,嘻嘻嘻。

 

例2,使用语法糖:

 1 import time
 2 
 3 def timer(func):
 4     # print('原本test的内存地址:', test)
 5     def deco():
 6         start_time = time.time()
 7         func()
 8         end_time = time.time()
 9         print('running time:',start_time-end_time)
10         print('func的内存地址:', func)
11         print('deco的内存地址:', deco)
12         print('最后test的内存地址:', test)
13     return deco
14 
15 @timer 
16 def test():
17     time.sleep(1)
18     print('in the test.')
19 
20 test()

 

输出结果如下:

1 in the test.
2 running time: 7.125407457351685
3 func的内存地址: <function test at 0x0000000003095EA0>
4 deco的内存地址: <function timer.<locals>.deco at 0x0000000003C63048>
5 最后test的内存地址: <function timer.<locals>.deco at 0x0000000003C63048>

 

运行到语法糖@timer(@timer即为test=timer(test))后,test函数的函数体被放于内存<function test at 0x0000000003095EA0>中,并传给形参func,

此时test=func=<function test at 0x0000000003095EA0>,但不知道为何无法打印出test原本的内存地址 = =

将deco函数体放于内存<function timer.<locals>.deco at 0x0000000003C63048>中,即deco=<function timer.<locals>.deco at 0x0000000003C63048>,

timer函数最后将deco返回,即test=deco=<function timer.<locals>.deco at 0x0000000003C63048>,偷梁换柱完成。

 

可能不够清楚(我也很绝望呀),多调试几次应该就能懂了,加油哦!

 

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯