前言
在Python中,装饰器是一种十分强大并且好用的语法,一些重复的代码使用装饰器语法的话能够使代码更容易理解及阅读。
因此在这里简单总结了一下Python中装饰器的几种用法以及需要注意的事情。
一、在装饰器中获取被装饰函数的参数
假设我们在开发web的时候,需要做反爬。要判断接口的访问来源我们就可以通过下面装饰器的方法来实现:
def mydecorator(func): def wrapped(*args, **kwargs): print("进入装饰器") if args[0]['header'] == 'spider': print("code: 400") return result = func(*args, **kwargs) return result return wrapped @mydecorator def request_page(request): print("一个访问请求") print("返回了response") if __name__ == '__main__': request = { 'data': 100, 'header': 'spider' } request_page(request)
在这个装饰器中,我们在装饰器中获取了request中的header参数,如果判断访问来源于爬虫,那么便给它返回一个400。
使用装饰器的写法等同于下面不使用装饰器的写法
def mydecorator(*args, **kwargs): print("进入函数") if args[0]['header'] == 'spider': print("code: 400") return False return True def request_page(request): if not mydecorator(request): return print("访问一个网页") print("得到了response") if __name__ == '__main__': request = { 'data': 100, 'header': 'spider' } request_page(request)
在只需要装饰一个函数的时候后面一种写法可能更优于装饰器的写法,但是在需要装饰很多个函数的时候,使用装饰器明显是更好的选择。
二、在装饰器获取函数的返回值
有的时候我们需要对函数的返回值做出判断,但又不想直接将判断写在函数里的时候,我们也可以使用装饰器来实现:
def mydecorator(func): def wrapped(*args, **kwargs): print("进入装饰器") result = func(*args, **kwargs) if result == 400: print("response is 400!") return False return True return wrapped @mydecorator def request_page(): print("访问一个网页") print("得到了response") return 200 if __name__ == '__main__': print(request_page())
三、给装饰器传入参数
在实际应用中,我们有时需要根据函数的执行状态来重复执行。例如在编写爬虫的时候,可能由于网络的原因会导致一些页面访问失败,这时我们就需要根据爬虫的返回结果进行重复请求。
def retry(MAXRETRY=3): def decorator(func): def wrapped(*args, **kwargs): print("进入装饰器") result = 0 retry = 1 while result != 200 and retry <= MAXRETRY: result = func(*args, **kwargs) print("重试第%s次" % retry) retry += 1 return result return wrapped return decorator @retry(5) def request_page(): print("访问一个网页") print("得到了response") return 400
在这里我们假设访问一个网页得到400的时候便重新请求。我们在retry装饰器里传了一个5,这表示我们希望重试的最大次数为5次,如果不传入这个值,那么它的默认重试次数则为3次。
在熟悉了基本装饰器的写法后,传参装饰器的写法也十分的好理解了。就是在外面多加了一层函数,用于传入参数。
四、装饰器文档的问题
我们都知道通过魔术方法__doc__可以获取我们写在代码中的文档,那么你是否知道使用装饰器后,会造成被包装函数的文档被装饰器的文档覆盖的问题呢。
def request_page(): ''' request_page 函数文档 :return: ''' print("访问一个网页") print("得到了response") if __name__ == '__main__': print(request_page.__doc__)
在上面对上面未使用装饰的代码使用__doc__方法的时候,我们得到的结果是:
In[3]: request_page.__doc__ Out[3]: '\n request_page 函数文档\n :return:\n '
这是我们理想中的结果!
但是当我们将上述函数使用装饰器装饰后:
def decorator(func): def wrapped(*args, **kwargs): ''' 装饰器文档 :param args: :param kwargs: :return: ''' print("进入装饰器") result = func(*args, **kwargs) return result return wrapped @decorator def request_page(): ''' request_page 函数文档 :return: ''' print("访问一个网页") print("得到了response")
我们再一次运行__doc__魔术方法的时候,得到的结果却是装饰器的内部文档:
In[4]: request_page.__doc__ Out[4]: '\n 装饰器文档\n :param args:\n :param kwargs:\n :return:\n ' In[5]: request_page.__name__ Out[5]: 'wrapped'
这个问题会使得我们的调试变得困难,也会使许多自动文档生成工具失去效果。
解决这个问题的最好办法就是使用 functools包的wraps()模块来将装饰器进行一个包装。
from functools import wraps def decorator(func): @wraps(func) def wrapped(*args, **kwargs): ''' 装饰器 :param args: :param kwargs: :return: ''' print("进入装饰器") result = func(*args, **kwargs) return result return wrapped @decorator def request_page(): ''' request_page 函数文档 :return: ''' print("访问一个网页") print("得到了response")
使用wraps将装饰器装饰后,这样我们的函数便能够保存它的一些重要数据了。
In[3]: request_page.__doc__ Out[3]: '\n request_page 函数文档\n :return:\n ' In[3]: request_page.__name__ Out[4]: 'request_page'
五、使用class的写法来编写装饰器
虽然大多数的装饰器都是通过函数的写法来实现的,但同样的可以通过类的写法来实现装饰器。
使用类的写法,我们可以实现一些使用函数写法不太好实现的需求。例如记录一个函数执行的次数
class Decorator(): def __init__(self,func): print('类初始化') self.func = func self.count = 0 def __call__(self, *args, **kwargs): print('进入装饰器') result = self.func(*args,**kwargs) self.count += 1 return result @Decorator def request_page(): ''' request_page :return: ''' print("访问一个网页") print("得到了response")
六、总结
装饰器是Python里比较高级的一种语法,这里只是介绍了它的几种使用技巧,以及需要注意的问题。借用金庸先生的话,“武功无高低,修为有深浅”。想要更加灵活的使用装饰器,深入理解它的原理,我们在平时还是需要加强基本功的学习!
好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对的支持。
免责声明:本站文章均来自网站采集或用户投稿,网站不提供任何软件下载或自行开发的软件! 如有用户或公司发现本站内容信息存在侵权行为,请邮件告知! 858582#qq.com
稳了!魔兽国服回归的3条重磅消息!官宣时间再确认!
昨天有一位朋友在大神群里分享,自己亚服账号被封号之后居然弹出了国服的封号信息对话框。
这里面让他访问的是一个国服的战网网址,com.cn和后面的zh都非常明白地表明这就是国服战网。
而他在复制这个网址并且进行登录之后,确实是网易的网址,也就是我们熟悉的停服之后国服发布的暴雪游戏产品运营到期开放退款的说明。这是一件比较奇怪的事情,因为以前都没有出现这样的情况,现在突然提示跳转到国服战网的网址,是不是说明了简体中文客户端已经开始进行更新了呢?
更新日志
- 雨林唱片《赏》新曲+精选集SACD版[ISO][2.3G]
- 罗大佑与OK男女合唱团.1995-再会吧!素兰【音乐工厂】【WAV+CUE】
- 草蜢.1993-宝贝对不起(国)【宝丽金】【WAV+CUE】
- 杨培安.2009-抒·情(EP)【擎天娱乐】【WAV+CUE】
- 周慧敏《EndlessDream》[WAV+CUE]
- 彭芳《纯色角3》2007[WAV+CUE]
- 江志丰2008-今生为你[豪记][WAV+CUE]
- 罗大佑1994《恋曲2000》音乐工厂[WAV+CUE][1G]
- 群星《一首歌一个故事》赵英俊某些作品重唱企划[FLAC分轨][1G]
- 群星《网易云英文歌曲播放量TOP100》[MP3][1G]
- 方大同.2024-梦想家TheDreamer【赋音乐】【FLAC分轨】
- 李慧珍.2007-爱死了【华谊兄弟】【WAV+CUE】
- 王大文.2019-国际太空站【环球】【FLAC分轨】
- 群星《2022超好听的十倍音质网络歌曲(163)》U盘音乐[WAV分轨][1.1G]
- 童丽《啼笑姻缘》头版限量编号24K金碟[低速原抓WAV+CUE][1.1G]