Python装饰器decorator用法实例


Posted in Python onNovember 10, 2014

本文实例讲述了Python装饰器decorator用法。分享给大家供大家参考。具体分析如下:

1. 闭包(closure)

闭包是Python所支持的一种特性,它让在非global scope定义的函数可以引用其外围空间中的变量,这些外围空间中被引用的变量叫做这个函数的环境变量。环境变量和这个非全局函数一起构成了闭包。

def outer(x):

    y = [1,2,3]

    def inner():

        print x

        print y

    return inner

x = 5    #这个x没有被引用

f = outer(2)

f()

print f.__closure__   #函数属性__closure__存储了函数的环境变量 def entrance(func):
 = 5    #这个x没有被引用f = outer(2)f()print f.__closure__   #函数属性__closure__存储了函数的环境变量 def entrance(func):
x和y都是属于函数outer命名空间的,在inner中被引用,当outer函数退出后,outer的命名空间不存在了,但是inner依然维护了其定义时候对其外部变量x,y的连接。
程序输出:

2

[1, 2, 3]

(, )

装饰器是一个可调用对象(a callable),在Python中,函数是对象,当然也是可调用的,所以装饰器可以是一个函数,我们称其为函数装饰器。
这个可调用对象以一个函数作为参数,闭且返回另一个函数(来替换参数那个函数)。
比如:

def entrance(func):

     def inner():

         print "inside function :", func.__name__

         func()

     return inner

 
entrance是一个装饰器,它是一个函数,它可以接收一个函数func作为参数,返回了另一个函数inner。
那为什么叫装饰器了,在返回函数inner()的内部,调用了func(),而且还作了额外的操作,相当于“装饰”了函数func。
那如何使用装饰器?
def fun1():

    pass

fun1 = entrance(fun1)

def fun2():

    pass

fun2 = entrance(fun2)

 
fun1,fun2的名字都没有变,但是通过调用函数装饰器entrance(),它们已经指向了另一个函数inner(),“装饰了”自己。

@操作符

Python提供的@符号,实质上就是上面做的,对一个函数名进行从新赋值,是语法上的技巧。所以上面的代码等价于

@entrance

def fun1():

    pass

@entrance

def fun2():

    pass

 
2. 装饰器的用途

从这个刻意构造的很简单的例子,可以看出装饰器的意义,如果一个函数需要一个功能,如果这个功能可以被使用在很多函数上,或是函数并不是自己实现,那可以写个装饰器来实现这些功能。
上面的装饰器entrance,装饰一个函数后,函数被调用时会打印出这个函数的名字。
但是有一个问题,这个装饰器从功能上看,是要应该可以用来装饰任何函数,但是如果我们用它来装饰了一个带参数的函数

@entrance

 def fun3(x):

     pass

只要不调用fun3,这三行代码是不会让Python解释器报错的,因为我们已经知道,它等价于:
def fun3(x):

    pass

fun3 = entrance(fun3)

 
我们定义了一个带参的函数fun3,然后把fun3指向了另一个函数inner(),当然不会有什么错。
 
但是,当我们使用fun3时,我们肯定会按照它定义时的样子去使用它,给它传入一个参数。
>>>fun3(1)
这里就会出错了,看看解释器怎么报错的

Traceback (most recent call last):
File “decorator.py”, line 23, in 3water.com <module>
fun3(1)
TypeError: inner() takes no arguments (1 given)

当然我们已经很容易知到为什么会这样报错了,fun3已经不是指向它定义时那个函数了,它现在指向了”inner()”,而inner是没有参数的,当然会出错。
那怎么解决呢?
修改一下inner()的定义,让它可以就收任意个参数就可以了。

def entrance(func):

     def inner(*args, **kvargs):

         print "inside function : ", func.__name__

         func(*args, **kvargs)

     return inner

现在,给inner传任意个参数都不会出错了,也就是entrance可以被用来装饰任何一个函数了。

3. 写个装饰器logger

一个函数被调用时,在日志里记录其名称和被调用的实际参数

def logger(func):

    def inner(*args, **kvargs):

        print  func.__name__, 'called, arguments: ', args, kvargs

        func(*args, **kvargs)

    return inner

希望本文所述对大家的Python程序设计有所帮助。

Python 相关文章推荐
python动态加载变量示例分享
Feb 17 Python
用Python编写一个每天都在系统下新建一个文件夹的脚本
May 04 Python
Python简单检测文本类型的2种方法【基于文件头及cchardet库】
Sep 18 Python
python如何使用unittest测试接口
Apr 04 Python
python多线程之事件Event的使用详解
Apr 27 Python
Django model序列化为json的方法示例
Oct 16 Python
用python拟合等角螺线的实现示例
Dec 27 Python
python GUI库图形界面开发之pyinstaller打包python程序为exe安装文件
Feb 26 Python
python3爬虫中多线程进行解锁操作实例
Nov 25 Python
Python模拟键盘输入自动登录TGP
Nov 27 Python
python 爬取腾讯视频评论的实现步骤
Feb 18 Python
python实现简单聊天功能
Jul 07 Python
python中list循环语句用法实例
Nov 10 #Python
python中MySQLdb模块用法实例
Nov 10 #Python
Python实现子类调用父类的方法
Nov 10 #Python
Python模仿POST提交HTTP数据及使用Cookie值的方法
Nov 10 #Python
Python类的多重继承问题深入分析
Nov 09 #Python
python查询mysql中文乱码问题
Nov 09 #Python
python刷投票的脚本实现代码
Nov 08 #Python
You might like
php实现的遍历文件夹下所有文件,编辑删除
2010/01/05 PHP
php输出xml格式字符串(用的这个)
2012/07/12 PHP
抛弃 PHP 代价太高
2016/04/26 PHP
完美解决php 导出excle的.csv格式的数据时乱码问题
2017/02/18 PHP
PHP中Notice错误常见解决方法
2017/04/28 PHP
php中错误处理操作实例分析
2019/08/23 PHP
PHP实现简单日历类编写
2020/08/28 PHP
js的压缩及jquery压缩探讨(提高页面加载性能/保护劳动成果)
2013/01/29 Javascript
JS 实现导航栏悬停效果
2013/09/23 Javascript
js取消单选按钮选中示例代码
2013/11/14 Javascript
jQuery中[attribute*=value]选择器用法实例
2014/12/31 Javascript
jQuery的one()方法用法实例
2015/01/19 Javascript
jQuery实现的图文高亮滚动切换特效实例
2015/08/10 Javascript
完美实现八种js焦点轮播图(下篇)
2020/04/20 Javascript
Vue2.0实现购物车功能
2017/06/05 Javascript
windows实现npm和cnpm安装步骤
2019/10/24 Javascript
vue与iframe之间的信息交互的实现
2020/04/08 Javascript
react结合bootstrap实现评论功能
2020/05/30 Javascript
Vue实现导航栏菜单
2020/08/19 Javascript
[01:05:32]DOTA2上海特级锦标赛主赛事日 - 3 败者组第三轮#1COL VS Alliance第一局
2016/03/04 DOTA
[01:59]深扒TI7聊天轮盘语音出处 1
2017/05/11 DOTA
python3.4用循环往mysql5.7中写数据并输出的实现方法
2017/06/20 Python
python去除字符串中的换行符
2017/10/11 Python
pandas.dataframe按行索引表达式选取方法
2018/10/30 Python
PyTorch实现重写/改写Dataset并载入Dataloader
2020/07/14 Python
python 怎样进行内存管理
2020/11/10 Python
html5 Canvas画图教程(9)—canvas中画出矩形和圆形
2013/01/09 HTML / CSS
使用spring mvc+localResizeIMG实现HTML5端图片压缩上传的功能
2016/12/16 HTML / CSS
美国汽配连锁巨头Pep Boys官网:轮胎更换、汽车维修服务和汽车零部件
2017/01/14 全球购物
英国健康和美容技术产品购物网站:CurrentBody
2019/07/17 全球购物
波兰珠宝品牌:YES
2019/08/09 全球购物
波兰运动鞋网上商店:Distance.pl
2020/07/30 全球购物
港湾网络笔试题
2014/04/19 面试题
Prototype中如何为一个元素添加一个方法
2014/12/08 面试题
2015领导干部廉洁自律工作总结
2015/07/23 职场文书
解决spring.thymeleaf.cache=false不起作用的问题
2022/06/10 Java/Android