浅析Python装饰器
1、什么是装饰器
在介绍装饰器之前,我们先来思考一个问题:使用Python语言进行程序设计时,如果我们想扩展一个函数的功能,一般会怎么做呢?
比如,有一个名为print_info函数,当前该函数内只做一些简单的打印操作,现在我们想扩展这个函数功能,如在发生错误时,我们将错误行号传入到该函数打印出来。
- def print_info():
- print("Hello World")
看到这个问题,我们的第一反应肯定是重新修改这个函数。在print_info函数增加一个参数,用于接收外部传入的错误行号,在函数内部增加打印行号信息语句。
- def print_info(err_line):
- print("Hello World")
- print("The Error Occurred Line Number: %d" %(err_line))
似乎,通过修改函数内容,也能实现我们的需求。但有没有这种可能呢,不直接进行修改print_info函数内容,通过其他的方式增加print_info函数的额外功能。答案肯定是有的,能实现这种功能的就是我们要讲的装饰器。
装饰器到底是什么呢?使用比较严谨的语言来描述:
- 装饰器本质上是一个Python函数,它可以让其他函数在不需要做任何代码变动的前提下增加额外功能,装饰器的返回值也是一个函数对象。它经常用于有切面需求的场景,比如:插入日志、性能测试、事务处理、缓存、权限校验等场景。
装饰器是解决这类问题的绝佳设计,有了装饰器,我们就可以抽离出大量与函数功能本身无关的雷同代码并继续重用。
概括的讲,装饰器的作用就是为已经存在的函数或对象添加额外的功能。
2、怎么写一个装饰器
我们先来看一段这样的代码
- 1 import sys
- 2
- 3 def log(fcn):
- 4 def wrapper(*argc, **kw):
- 5 #New Operations Are Added Here
- 6 if len(argc) != 1:
- 7 print('Illegal parameter')
- 8 return -1
- 9 print("The Error Occurred Line Number: %d" %(argc[0]))
- 10 return fcn()
- 11 return wrapper
- 12
- 13 def print_info():
- 14 print("Hello World")
- 15
- 16 print_info = log(print_info)
# 添加功能并保持原函数名不变
- 17
- 18 def main():
- 19 print_info(sys._getframe().f_lineno) # sys._getframe().f_lineno代表当前的行号
- 20
- 21 if __name__ == "__main__":
- 22 main()
log函数是一个“闭包”(关于什么是“闭包”可参考“浅析Python闭包”),该函数有一个参数fcn,返回值是内部实现的wrapper函数。
比较重要的一条语句print_info = log(print_info),这里通过对print_info变量赋值改变它原来指向的类型,起到了对print_info函数赋予了新功能。执行这句语句之后print_info变量不再表示前面定义的print_info函数,而是表示log函数内实现的wrapper函数,但原来定义的print_info函数仍然存在。
最终调用的print_info函数,其实调用的是log中的wrapper函数。由于“闭包”特性,虽然wrapper函数已经离开了创造它的环境log函数,但它仍然可以使用log函数中的自由变量。wrapper函数中,打印了传入的错误行号,并调用了传入的原定义的print_info函数,这样便做到了对定义的print_info函数内容做修改,调用print_info增加了额外功能。
运行结果
- The Error Occurred Line Number: 19 # 打印出新增的对应行号信息,在19行被调用,所以传入行号是19
- Hello World #原print_info函数执行内容
上面的log函数其实已经是一个装饰器了,它对原函数做了包装并返回了另外一个函数,额外添加了一些功能。因为这样写实在不太优雅,新版本Python中支持了@语法糖,我们可以把它使用起来。
下面代码等同于上面的写法。
- import sys
- def log(fcn):
- def wrapper(*argc, **kw):
- #New Operations Are Added Here
- if len(argc) != 1:
- print('Illegal parameter')
- return -1
- print("The Error Occurred Line Number: %d" %(argc[0]))
- return fcn()
- return wrapper
- @log
- def print_info():
- print("Hello World")
- #print_info = log(print_info)
- def main():
- print_info(sys._getframe().f_lineno)
- if __name__ == "__main__":
- main()
把@log放到print_info函数定义处,相当于执行了print_info = log(print_info)语句
3、完整装饰器的写法
前面写的装饰器都没有问题,但是还差最后一步,虽然装饰器装饰过的函数看上去名字没变,其实已经变了。
- def log(fcn):
- def wrapper(*args, **kw):
- print(fcn.__name__)
- return fcn()
- return wrapper
- @log
- def print_info():
- print('Hello World')
- def main():
- print_info()
- print(print_info.__name__)
- if __name__ == "__main__":
- main()
运行结果
- print_info
Hello World
wrapper
>>>
上面print_info函数经log装饰器装饰后,它的__name__属性已经从原来的‘print_info’变成了‘wrapper’
因为返回的那个wrapper()函数名字就是'wrapper',所以,需要把原始函数的__name__等属性复制到wrapper()函数中,否则,有些依赖函数签名的代码执行就会出错。
不需要在内部添加wrapper.__name__=fcn.__name__这样的代码,Python内置的functools.wraps就是干这个事的,所以,一个完整的decorator的写法如下:
- import functools
- def log(fcn):
- @functools.wraps(fcn)
- def wrapper(*args, **kw):
- print(fcn.__name__)
- return fcn()
- return wrapper
4、高阶一点装饰器
4.1 带参数装饰器
如果装饰器函数本身要传入参数,那么装饰器就会是这样的
- import sys
- def log(text):
- def wrapper(fcn)
- def inner_wrapper(*argc, **kw):
- #New Operations Are Added Here
- print(text)
- if len(argc) != 1:
- print('Illegal parameter')
- return -1
- print("The Error Occurred Line Number: %d" %(argc[0]))
- return fcn()
- return inner_wrapper
- return wrapper
- @log('With Parameter Decorator')
- def print_info():
- print("Hello World")
- #print_info = log(print_info)
- def main():
- print_info(sys._getframe().f_lineno)
- if __name__ == "__main__":
- main()
对于带参数装饰器,可以这么理解,当带参数的装饰器被装饰在某个函数上时,比如上述代码@log('With Parameter Decorator'),
log('With Parameter Decorator')其实是一个函数,会马上被执行,它返回的结果任是一个装饰器wrapper。
也就是下面代码其实等价于print_info = wrapper(print_info)
- @log('With Parameter Decorator')
- def print_info():
- print("Hello World")
4.2 装饰器装饰类方法
前面我们介绍的都是用装饰器装饰函数,装饰器同时也可以装饰类方法,下面来看如何用装饰器装饰类的方法
我们定义了名为simple_test的类,这个类有一个data属性和get_data方法。类里的实现非常简单,get_data方法只返回data属性值,并不进行其他操作。同时,我们还需定义了一个装饰器装饰类的get_data方法,装饰器中根据数据获取源来编写其中功能,比如我们这里选择从串口终端获取数据,直接修改装饰器的内容让其获取终端输入数据。
这样在设计一个类时,做到了将类中稳定的部分和经常要修改的部分独立开。类设计完成后,我们不需要更改类的源码,在使用时只需要根据需求修改类方法的装饰器就能实现我们的需求。
- def serial_data(fcn):
- def wrapper(self, *args, **kw):
- str = input('Please enter an integer: ')
- try:
- tmp = int(str)
- except ValueError:
- print('Invalid Value')
- return None
- self.data = tmp
- return fcn(self)
- return wrapper
- class simple_test(object):
- def __init__(self, _val = 0):
- self.data = _val
- @serial_data
- def get_data(self):
- return self.data
- def main():
- obj = simple_test()
- val = obj.get_data()
- if val is not None:
- print('get Data: %d' %(val))
- if __name__ == "__main__":
- main()
运行结果:
- Please enter an integer: 100
- get Data: 100
- >>>
浅析Python装饰器的更多相关文章
- 粗浅聊聊Python装饰器
浅析装饰器 通常情况下,给一个对象添加新功能有三种方式: 直接给对象所属的类添加方法: 使用组合:(在新类中创建原有类的对象,重复利用已有类的功能) 使用继承:(可以使用现有类的,无需重复编写原有类进 ...
- 关于python装饰器
关于python装饰器,不是系统的介绍,只是说一下某些问题 1 首先了解变量作用于非常重要 2 其次要了解闭包 def logger(func): def inner(*args, **kwargs) ...
- python装饰器通俗易懂的解释!
1.python装饰器 刚刚接触python的装饰器,简直懵逼了,直接不懂什么意思啊有木有,自己都忘了走了多少遍Debug,查了多少遍资料,猜有点点开始明白了.总结了一下解释得比较好的,通俗易懂的来说 ...
- Python 装饰器学习
Python装饰器学习(九步入门) 这是在Python学习小组上介绍的内容,现学现卖.多练习是好的学习方式. 第一步:最简单的函数,准备附加额外功能 1 2 3 4 5 6 7 8 # -*- c ...
- python 装饰器修改调整函数参数
简单记录一下利用python装饰器来调整函数的方法.现在有个需求:参数line范围为1-16,要求把9-16的范围转化为1-8,即9对应1,10对应2,...,16对应8. 下面是例子: def fo ...
- python 装饰器学习(decorator)
最近看到有个装饰器的例子,没看懂, #!/usr/bin/python class decorator(object): def __init__(self,f): print "initi ...
- Python装饰器详解
python中的装饰器是一个用得非常多的东西,我们可以把一些特定的方法.通用的方法写成一个个装饰器,这就为调用这些方法提供一个非常大的便利,如此提高我们代码的可读性以及简洁性,以及可扩展性. 在学习p ...
- 关于python装饰器(Decorators)最底层理解的一句话
一个decorator只是一个带有一个函数作为参数并返回一个替换函数的闭包. http://www.xxx.com/html/2016/pythonhexinbiancheng_0718/1044.h ...
- Python装饰器由浅入深
装饰器的功能在很多语言中都有,名字也不尽相同,其实它体现的是一种设计模式,强调的是开放封闭原则,更多的用于后期功能升级而不是编写新的代码.装饰器不光能装饰函数,也能装饰其他的对象,比如类,但通常,我们 ...
随机推荐
- 蓝桥杯——四数平方(2016JavaB第7题)
四数平方(16JavaB7) 四平方和定理,又称为拉格朗日定理: 每个正整数都可以表示为至多4个正整数的平方和. 如果把0包括进去,就正好可以表示为4个数的平方和. 比如: 5 = 0^2 + 0^2 ...
- activiti数据表介绍
activiti6.0数据库介绍 Acitiviti6.0数据库中一共有28张表,表的命名都是以ACT_开头的.第二部分是一个两个字符用例表的标识. 用于自我学习记录,后期不定期更新~~~ 数据库描述 ...
- 使用@RequestBody注解获取Ajax提交的json数据
最近在学习有关springMVC的知识,今天学习如何使用@RequestBody注解来获取Ajax提交的json数据内容. Ajax部分代码如下: 1 $(function(){ 2 $(" ...
- 19_B门长时曝光APP
知识很基础-- 前几天买了个单反,特别想拍B门长时间曝光的效果.后来想想不如自己写个APP,实现屏幕背景的随机颜色以及全屏显示文字. 先上图: 这两张图片的左侧都很亮,这是因为APP里面忘记把&quo ...
- angular中datetime-local属性使用ng-model报错
项目需求是将年月日格式更改为年月日时分的格式展示,翻遍了整个项目没找到符合的组件,自己现敲一个也来不及,只好直接使用原生自带的组件--datetime-local.之前都是用的vue写项目,第一次接触 ...
- 倾斜摄影实景三维在智慧工厂 Web 3D GIS 数字孪生应用
数字化推动钢铁工业转型升级 数字时代,随着数字地球,数字中国,数字工厂等数字化建设的不断深入,以地理信息系统(Geographic Information System, GIS)为基础,融合大数 ...
- PyQt(Python+Qt)学习随笔:QTreeWidgetItem项的子项索引、删除子项的方法
老猿Python博文目录 专栏:使用PyQt开发图形界面Python应用 老猿Python博客地址 树型部件QTreeWidget中的QTreeWidgetItem项: 获取子项索引 可通过index ...
- spring整合mybatis01
title: spring整合mybatis01 date: 2020-03-09 19:47:40 tags:整合的第一种方式 spring整合mybatis 1.mybatis回顾 mybatis ...
- 返回sourceString 中出现的第一个 searchString 的索引
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- 题解-CF708D Incorrect Flow
题面 CF708D Incorrect Flow 给一张网络流图,可能有流量不守恒或者流量超过容量的情况,求最少的将某条边流量或容量 \(\pm 1\) 的操作次数使得网络流图正确. 数据范围:\(1 ...