Python 装饰器
Posted 烟草的香味
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Python 装饰器相关的知识,希望对你有一定的参考价值。
首先
data:image/s3,"s3://crabby-images/a194f/a194f3c8b6b0c7435b8bec9d391b831278ddda0c" alt="Python 装饰器"
data:image/s3,"s3://crabby-images/dd618/dd61851925d54373b75c32e87fec43aa402a8c53" alt="Python 装饰器"
需求来了
有如下几个封装好的函数供调用:
data:image/s3,"s3://crabby-images/78dc9/78dc954d78678b5ead5f1c3d2d258a0fd4071e26" alt="Python 装饰器"
现在需要在每个函数执行前进行日志记录:
第一个方案
data:image/s3,"s3://crabby-images/c7840/c784095ea6664fdadcef54e430d54db3db48c88a" alt="Python 装饰器"
修改每个函数,添加日志记录的代码
但这样显然不太好,存在大量的重复代码,可以将重复代码封装为一个方法
第二个方案
data:image/s3,"s3://crabby-images/d88b8/d88b8f76ba8101393e5ae2320169ae78cb34067c" alt="Python 装饰器"
这样的确是比第一个方案好多了,但是不符合开闭原则,即现有的代码不要去修改,而在基础的功能上进行二次开发
第三个方案
data:image/s3,"s3://crabby-images/a6ed3/a6ed32043824f09b4ef69c7b76f2b2d399eeb803" alt="Python 装饰器"
这样的确很好,但是在调用的时候太麻烦了,而且函数名字也变了,以前写好的代码全部都要修改,想办法让函数名不变就好了
第四个方案
data:image/s3,"s3://crabby-images/5b5d9/5b5d9dcef2e506f45bd6112eb29b3f6a0367de23" alt="Python 装饰器"
这样就又好点了,在增加功能而不修改代码的前提下,也保证了函数名不变
这就是装饰器的功能了,装饰器和上边的代码实现的功能相同,只不过用了语法糖
使用装饰器的方案
data:image/s3,"s3://crabby-images/52f4e/52f4ea7423c74bc7addaf37528d8fb4db1b77af6" alt="Python 装饰器"
装饰器原理
上述代码在执行@verify时,进行了如下操作:
执行verify函数 ,并将 @verify 下面的函数作为verify函数的参数
即@verify 等价于 verify(f1) , 其内部执行代码如下:
f1 = verify(f1)
如此便为装饰器
小小装饰器
data:image/s3,"s3://crabby-images/4c849/4c849e7abffb696026ad9818efb63214df1847ca" alt="Python 装饰器"
输出结果:
data:image/s3,"s3://crabby-images/da489/da489384e7f1a61667e655a6267c4d5145093951" alt="Python 装饰器"
装饰器的常用功能
- 引入日志
- 函数执行时间统计
- 执行函数前预备处理
- 执行函数后清理功能
- 权限校验等场景
- 缓存
装饰器实例
1.无参数的函数
data:image/s3,"s3://crabby-images/965ec/965ec19706e17ef954e96077673d0384767abf4f" alt="Python 装饰器"
2.被装饰的函数有参数
data:image/s3,"s3://crabby-images/784dd/784dda0d1d8cd51a007d2f3d9c8061f8ac5b8461" alt="Python 装饰器"
3.被装饰的函数有不定长参数
data:image/s3,"s3://crabby-images/85c01/85c017f7e33feef6228335f274acef9aa6e6e206" alt="Python 装饰器"
4.装饰器中的return
data:image/s3,"s3://crabby-images/9b094/9b0943c035c4d2ce1723afff2a24a65eaeb6bf5f" alt="Python 装饰器"
执行结果
data:image/s3,"s3://crabby-images/3ee18/3ee187b76af8137c68e2f5ed00046b8ef7f32e9f" alt="Python 装饰器"
如果修改装饰器为 return func(),则运行结果:
data:image/s3,"s3://crabby-images/abc5e/abc5e9f4c711fbd9ccf5b98687ccc215296d13a1" alt="Python 装饰器"
所以,一般情况下为了让装饰器更通用,可以有return
5.装饰器带参数,在原有装饰器的基础上,设置外部变量
data:image/s3,"s3://crabby-images/3c904/3c904e09120f516f5a01d552601414a40f8d609d" alt="Python 装饰器"
运行结果为:
data:image/s3,"s3://crabby-images/dac82/dac82a73c845b7250453ade92e1343fe64769f06" alt="Python 装饰器"
可以理解为:
foo()==timefun_arg("itcast")(foo)()
以上是关于Python 装饰器的主要内容,如果未能解决你的问题,请参考以下文章