流畅的python--装饰器
装饰器:以某种方式增强函数。
两大特性:1、可以将被装饰的函数替换成其他函数。
2、在加载模块时立即执行。
案例1
def make_avarage():
count=0
total=0
def averager(new_value):
count+=1
total+=new_value
return total/count
return averager
这里会报错,averager里面是一个新的 count、total变量
案例2
使用nolocal作用:将变量标记为自由变量,
def make_avarage():
count=0
total=0
def averager(new_value):
nonlocal count,total
count+=1
total+=new_value
return total/count
return averager
案例3
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
"""
Created on Sun May 3 23:27:22 2020 @author: root
""" promos=[]
def promotion(promo_func):
promo.append(promo_func)
return promo_func @promotion
def FidelityPromo(order):
return order.total()*0.05 if order.customer.fidelity>=1000 else 0 @promotion
def BulkItemPromo(order):
discount = 0
for item in order.cart:
if item.quantity>=20:
discount+=item.total()*0.1
return discount @promotion
def LargeOrderPromo(order):
discount_items = {item.product for item in order.cart}
if len(discount_items)>=10:
return order.total()*0.07
return 0 def best_promo(order):
return max(promo(order) for promo in promos)
二、@property:方法伪装属性,方法返回值及属性值,被装饰方法不能有参数,必须实例化后调用,类不能调用
#@ property
#将一个方法伪装成属性,被修饰的特性方法,内部可以实现处理逻辑,但对外提供统一的调用方式,实现一个实例属性的
getter
setter
d elete
三种方法的内部逻辑,具体含义看示
class Data:
def __init__(self):
self.number = 123
@property
def operation(self):
return self.number
@operation.getter
def operation(self):
return self.number
@operation.setter
def operation(self, number):
self.number = number
@operation.deleter
def operation(self):
del self.number
案例1----普通写法
class Student(object):
def get_score(self):
return self._score
def set_score(self,value):
if not isinstance(value,int):
raise ValueError('pppppp')
if value < 0 or value > 100:
raise ValueError('score must between 0 ~ 100!')
self._score = value
Python内置的@property装饰器就是负责把一个方法变成属性调用
案例1----装饰器写法
class Student(object):
@property
def score(self):
return self._score
@score.setter
def score(self, value):
if not isinstance(value, int):
raise ValueError('pppppp')
if value < 0 or value > 100:
raise ValueError('score must between 0 ~ 100!')
self._score = value
s=Student()
s.score=60
s.score=9999
print (s.score)
三、类装饰器:类装饰器使用地方较少,核心是通过复写类的回调方法call.
装饰器应用场景:
1、引入日志
2、函数执行时间统计
3、执行函数前预备处理
执行函数后清理功能
权限校验等场景
缓存
方法:__call__
class Add:
def __init__(self, fn):
print("初始化")
self.num = 44
self.fn = fn
def __call__(self, *args,**kwargs):
print("类装饰器开始工作")
return self.fn(self.num)
@Add
def test(n):
return 4+n
if __name__ == '__main__':
num = test()
print(num)
结果:
<function test at 0x7f95f883bc20>
类装饰器开始工作
44
48
案例1
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
"""
Created on Mon May 4 22:11:44 2020 @author: root
简单注册表
""" funcs = []
def register(func):
funcs.append(func)
return func @register
def a():
return 3 @register
def b():
return 5 def c():
return 8 # 访问结果
result = [func() for func in funcs]
print(funcs)
print(result) # [<function a at 0x7f95f8837050>, <function b at 0x7f95f8837290>]
# [3, 5]
案例2
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
"""
Created on Mon May 4 22:14:37 2020
@author: root
注册表隔离(使用类的不同实例)
"""
class Registry(object):
def __init__(self):
self._funcs=[]
def register(self,func):
self._funcs.append(func)
def run_all(self):
return [func() for func in self._funcs]
r1=Registry()
r2=Registry()
@r1.register
def a():
return 3
@r2.register
def b():
return 5
案例3
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
"""
Created on Mon May 4 22:30:41 2020 @author: root 用户验证
""" from functools import wraps class User(object):
def __init__(self, username, email):
self.username = username
self.email = email class AnonymousUser(object):
def __init__(self):
self.username = self.email = None def __nonzero__(self): # 将对象转换为bool类型时调用
return False def requires_user(func):
@wraps(func)
def inner(user, *args, **kwargs): # 由于第一个参数无法支持self, 该装饰器不支持装饰类
if user and isinstance(user, User):
return func(user, *args, **kwargs)
else:
raise ValueError("非合法用户")
return inner a=User('teddy', '120@163.com')
print (a)
案例3
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
"""
Created on Mon May 4 22:36:47 2020 @author: root 异常捕获
""" import json
from functools import wraps class Error1(Exception):
def __init__(self, msg):
self.msg = msg
def __str__(self):
return self.msg def json_output(func):
@wraps(func)
def inner(*args, **kwargs):
try:
result = func(*args, **kwargs)
except Error1 as ex:
result = {"status": "error", "msg": str(ex)}
return json.dumps(result)
return inner # 使用方法
@json_output
def error():
raise Error1("该条异常会被捕获并按JSON格式输出")
案例3
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
"""
Created on Mon May 4 22:40:47 2020 @author: root 日志管理
"" import time
import logging
from functools import wraps def logged(func):
@wraps(func)
def inner(*args, **kwargs): # *args可以装饰函数也可以装饰类
start = time.time()
result = func(*args, **kwargs)
exec_time = time.time() - start
logger = logging.getLoger("func.logged")
logger.warning("{} 调用时间:{:.2} 执行时间:{:.2}s 结果:{}".format(func.__name__, start, exec_time, result))
资源清理#import pymysql class DBConnection(object):
def __init__(self, *args, **kwargs):
self.args,self.kwargs = args, kwargs def __enter__(self):
self.conn = pymysql.connect(*args, **kwargs)
return self.conn.cursor() def __exit__(self, exc_type, exc_instance, trackback):
self.conn.close()
案例来源:https://www.cnblogs.com/superhin/p/11454823.html
流畅的python--装饰器的更多相关文章
- Python装饰器:套层壳我变得更强了
Python装饰器:套层壳我变得更强了 Python装饰器:套层壳我变得更强了 关于作用域和闭包可以聊点什么? 什么是作用域 什么是闭包 装饰器:套层壳我变得更强了 参考资料 昨天阅读了<Pyt ...
- 关于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装饰器由浅入深
装饰器的功能在很多语言中都有,名字也不尽相同,其实它体现的是一种设计模式,强调的是开放封闭原则,更多的用于后期功能升级而不是编写新的代码.装饰器不光能装饰函数,也能装饰其他的对象,比如类,但通常,我们 ...
- Python装饰器与面向切面编程
今天来讨论一下装饰器.装饰器是一个很著名的设计模式,经常被用于有切面需求的场景,较为经典的有插入日志.性能测试.事务处理等.装饰器是解决这类问题的绝佳设计,有了装饰器,我们就可以抽离出大量函数中与函数 ...
随机推荐
- 免费版对象存储【minIO】CentOS部署实践记录 2021
好久没写,记录一下 1.背景 之前一直用的七牛,不过是收费的,然后有些定制化需求,可能比较看重预算,然后就有了这篇开源方式:minio 2.简介 官方文档:http://docs.minio.org. ...
- 【NX二次开发】Block UI 超级截面
属性说明 属性 类型 描述 常规 BlockID String 控件ID Enable Logical 是否可操作 Group ...
- 618技术特辑(三)直播带货王,“OMG买它”的背后,为什么是一连串技术挑战?
[本期推荐]为什么一到大促,我们的钱包总是被掏空?是大家自制力不够,还是电商平台太会读懂人心,从技术维度,抽丝剥茧一探究竟. 摘要:动辄几十上百万人同时在线的直播间,让所有人能同时公平的去抢购,并且还 ...
- 【模拟8.11】将军令(贪心&&树形DP)
只看45分的话,是树形DP....(当然也有能拿到70分+的大佬) 40分: 只考虑k==1的情况,树形DP 所以每个节点可能被父亲,自己,儿子控制 设f[MAXN][3],0表示儿子,1表示自己,2 ...
- NOIP模拟测试28「阴阳·虎·山洞」
写这几个题解我觉得我就像在按照官方题解抄一样 阴阳 题解 将题目中给的阴阳看作黑色和白色 首先我们观察到最后生成图中某种颜色必须是竖着单调递增或竖着单调递减 类似这样 否则不满足这个条件 但合法染色方 ...
- 详解详解Java中static关键字和final关键字的功能
摘要:static关键字和final关键字是Java语言的核心,深入理解他们的功能非常重要. 本文分享自华为云社区<Java: static关键字与final关键字>,原文作者:唐里 . ...
- php 安装 yii 报错: phpunit/phpunit 4.8.32 requires ext-dom *
php 安装 yii 报错: phpunit/phpunit 4.8.32 requires ext-dom * 我的版本是7.0,以7.0为例演示. 先装这两个拓展试试: sudo apt-get ...
- [HNOI2006]公路修建问题题解
题目 题目描述 OI island是一个非常漂亮的岛屿,自开发以来,到这儿来旅游的人很多.然而,由于该岛屿刚刚开发不久,所以那里的交通情况还是很糟糕.所以,OIER Association组织成立了, ...
- bcprov-jdk15-145.rar
javapdf文件操作,加密包 bcprov-jdk15-145.rar https://files.cnblogs.com/files/blogs/692137/bcprov-jdk15-145.r ...
- ansible 配置详解
ansible 安装方式 ansible安装常用两种方式,yum安装和pip程序安装.下面我们来详细介绍一下这两种安装方式. 使用 pip(python的包管理模块)安装 首先,我们需要安装一个pyt ...