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抓取网页正文的源码
Jun 11 Python
python实现支持目录FTP上传下载文件的方法
Jun 03 Python
简单实现python数独游戏
Mar 30 Python
Python Tkinter模块实现时钟功能应用示例
Jul 23 Python
Window环境下Scrapy开发环境搭建
Nov 18 Python
解决pycharm每次新建项目都要重新安装一些第三方库的问题
Jan 17 Python
python实现多层感知器
Jan 18 Python
将pip源更换到国内镜像的详细步骤
Apr 07 Python
python如何实现异步调用函数执行
Jul 08 Python
selenium与xpath之获取指定位置的元素的实现
Jan 26 Python
python中Matplotlib绘制直线的实例代码
Jul 04 Python
python数字转对应中文的方法总结
Aug 02 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 cookis创建实现代码
2009/03/16 PHP
php生成局部唯一识别码LUID的代码
2012/10/06 PHP
Yii框架中memcache用法实例
2014/12/03 PHP
laravel-admin解决表单select联动时,编辑默认没选上的问题
2019/09/30 PHP
jquery ajax例子返回值详解
2012/09/11 Javascript
Jquery图形报表插件 jqplot简介及参数详解
2012/10/10 Javascript
javascript 全选与全取消功能的实现代码
2012/12/23 Javascript
用表格输出1-1000之间的数字实现代码(附特效)
2013/04/21 Javascript
火狐下table中创建form导致两个table之间出现空白
2013/09/02 Javascript
Nodejs使用mysql模块之获得更新和删除影响的行数的方法
2014/03/18 NodeJs
jQuery实现动态添加和删除一个div
2015/08/12 Javascript
JS实现带关闭功能的阿里妈妈网站顶部滑出banner工具条代码
2015/09/17 Javascript
AngularJS 模块化详解及实例代码
2016/09/14 Javascript
微信小程序如何获取用户手机号
2018/01/26 Javascript
vue-cli项目配置多环境的详细操作过程
2018/10/30 Javascript
Vue 事件处理操作实例详解
2019/03/05 Javascript
vue3.0 项目搭建和使用流程
2021/03/04 Vue.js
[54:24]Optic vs TNC 2018国际邀请赛小组赛BO2 第二场
2018/08/18 DOTA
python 域名分析工具实现代码
2009/07/15 Python
Centos下实现安装Python3.6和Python2共存
2018/08/15 Python
Python3基础教程之递归函数简单示例
2019/06/07 Python
在pytorch中对非叶节点的变量计算梯度实例
2020/01/10 Python
基于python实现图片转字符画代码实例
2020/09/04 Python
python3实现语音转文字(语音识别)和文字转语音(语音合成)
2020/10/14 Python
地球上最先进的胡子和头发修剪器:Bevel
2018/01/23 全球购物
表演方阵解说词
2014/02/08 职场文书
领导党性分析材料
2014/02/15 职场文书
金融管理专业毕业生求职信
2014/03/12 职场文书
yy婚礼主持词
2014/03/14 职场文书
放飞中国梦演讲稿
2014/04/23 职场文书
本科毕业生自荐信
2014/06/02 职场文书
幼儿园教师师德表现自我评价
2015/03/05 职场文书
小学作文指导之如何写人?
2019/07/08 职场文书
Redis如何实现分布式锁
2021/08/23 Redis
MySQL利用UNION连接2个查询排序失效详解
2021/11/20 MySQL
Win11电源已接通但未充电怎么办?Win11电源已接通未充电的解决方法
2022/04/05 数码科技