Python的装饰器用法学习笔记
作者:cangmean 时间:2021-05-06 03:03:36
在python中常看到在定义函数是使用@func. 这就是装饰器, 装饰器是把一个函数作为参数的函数,常常用于扩展已有函数,即不改变当前函数状态下增加功能.
def run():
print "I'm run."
我有这么一个函数, 我想知道这个函数什么时候开始什么时候结束. 我应该这么写
def run():
print time.ctime()
print "I'm run."
print time.ctime()
但是如果不允许修改函数的话就需要装饰器了
def count(func):
def wrapper():
print time.ctime()
ret = func()
print time.ctime()
return ret
return wrapper
@count
def run():
print "I'm run."
# print '2015-4-10'
eg:
def now():
print '2015-4-10'
f = now
f()
函数有一个__name__ 对象 可通过 dir(func) func为定义的函数名
now.__name__ # print 'now'
f.__name__ # print 'now'
print f # print '<function now at 0x000000000213A908>'
print now # print '<function now at 0x000000000213A908>'
我们通过装饰器打印log日志
def log(func):
def wrapper(*args, **kwargs):
print "call %s()" % func.__name__
return func(*args, **kwargs)
return wrapper
@log
def now():
print '2015-4-10'
now() # print 'call now()'
其实装饰器修饰函数相当于, now = log(now) 也就是装饰器函数把被修饰的函数当参数后赋给同名的变量
functools.wraps 函数
当我们使用了装饰器后now的__name__值发生了改变
# 没有使用前
now.__name__ # print 'now'
# 使用后
now.__name__ # print 'wrapper'
当我们使用装饰器前,now.__name__使用的是当前now函数,但使用后 now这个函数其实是 log(now) 也就是log函数的返回值也就是被包裹的wrapper. 解决方法是functools.wraps函数.
装饰闭包, 使用前得调用 import functools
def log(func):
@functools.wraps(func)
def wrapper(*args, **kwargs):
...
带参数的装饰器
如果decorator需要传入参数, 那就需要在写一个返回decorator的高阶函数. 写出来更复杂.
def login(level):
def _deco(func):
def wrapper(*args, **kwargs):
if level >= 5:
print '用户 VIP 等级 %d' % int(level-5)
else:
print '用户 * 丝 等级 %d' % abs(level-5)
return func(*args, **kwargs)
return wrapper
return _deco
@login(5)
def user(username):
print 'welcome, %s' % username
# 用户vip 等级0
# welcome, mink
user('mink')
带参数的decorator等于func = 装饰器函数(装饰器参数)(func)
装饰器类
通过类的__call__可以想使用函数一样使用类
class A(object):
def __init__(self, func):
self.func = func
def __call__(self):
return self.func() ** 2
@A
def foo():
return 10
print foo() # print 100
标签:Python,装饰器
![](/images/zang.png)
![](/images/jiucuo.png)
猜你喜欢
Python编程django实现同一个ip十分钟内只能注册一次
2023-01-18 17:01:55
python 自动去除空行的实例
2021-04-12 08:46:04
Pygame鼠标进行图片的移动与缩放案例详解
2023-08-12 15:18:58
![](https://img.aspxhome.com/file/2023/5/97525_0s.gif)
Python机器视觉之基于OpenCV的手势检测
2021-06-12 10:54:11
![](https://img.aspxhome.com/file/2023/9/103359_0s.jpg)
Python使用回溯法子集树模板获取最长公共子序列(LCS)的方法
2021-04-14 04:55:49
![](https://img.aspxhome.com/file/2023/5/104885_0s.jpg)
Python创建xml文件示例
2023-03-08 22:41:45
Tesserocr库的正确安装方式
2023-12-26 17:19:34
![](https://img.aspxhome.com/file/2023/3/124363_0s.png)
一文学会利用python解决文章付费限制问题
2021-04-09 08:23:51
![](https://img.aspxhome.com/file/2023/7/82537_0s.jpg)
利用Python+OpenCV三步去除水印
2021-09-17 10:32:30
![](https://img.aspxhome.com/file/2023/7/80617_0s.jpg)
线程安全及Python中的GIL原理分析
2022-09-06 11:16:20
三种SQL分页法
2010-05-07 11:03:00
如何基于Python和Flask编写Prometheus监控
2021-03-21 13:37:13
Python设计模式编程中的备忘录模式与对象池模式示例
2023-02-06 05:48:43
解决Scrapy安装错误:Microsoft Visual C++ 14.0 is required...
2023-03-06 19:48:57
Python3.x爬虫下载网页图片的实例讲解
2022-11-25 23:24:07
![](https://img.aspxhome.com/file/2023/5/76735_0s.jpg)
python入门学习之自带help功能初步使用示例
2021-05-27 17:07:28
如何列举Error的所有对象?
2010-01-12 20:01:00
asp随机提取access数据库记录的几种方法
2007-09-06 19:42:00
Python爬虫文件下载图文教程
2023-11-18 23:59:03
![](https://img.aspxhome.com/file/2023/5/86195_0s.png)
Python基础中所出现的异常报错总结
2023-07-14 05:33:47
![](https://img.aspxhome.com/file/2023/0/107490_0s.png)