详解Python装饰器的四种定义形式
作者:北极象 发布时间:2022-06-20 13:35:04
标签:Python,装饰器
前言
装饰器(decorator)在Python框架中扮演着重要角色,是Python中实现切面编程(AOP)的重要手段。
aspect-oriented programming (AOP) ,在不改变代码自身的前提下增加程序功能
不改变代码自身,但需要在函数和类头上加一个标注(annotation),这个标注在Python里叫装饰器,在java里叫注解。
在Python里,一共有四种组合形式。下面一一举例。
用函数装饰函数
采用一个函数定义装饰器:
def decorate(f):
def wrapper(*args):
return f(*args)*2
return wrapper
然后作用在一个函数上:
@decorate
def add(a, b):
return a + b
测试一下效果:
def test_decorate():
sum = add(3, 5)
assert sum == 16
用函数装饰一个类
这里通过装饰器实现单例模式:
def singleton(cls):
instances = {}
def wrapper(*args, **kwargs):
if cls not in instances:
instances[cls] = cls(*args, **kwargs)
return instances[cls]
return wrapper
使用该装饰器:
@singleton
class MyClass:
def method(self):
pass
于是,当你定义多个对象时,返回的是同一实例:
obj = MyClass() # creates a new instance
obj2 = MyClass() # returns the same instance
obj3 = MyClass() # returns the same instance
...
用类定义装饰器,然后装饰一个函数
先采用类定义一个装饰器:
class Star:
def __init__(self, n):
self.n = n
def __call__(self, fn):
@wraps(fn)
def wrapper(*args, **kwargs):
result = fn(*args, **kwargs)
return result
return wrapper
再作用在一个函数上:
@Star(5)
def add(a, b):
return a + b
主要是在类中实现__call__方法。上面例子也可以简化:
class MyDecorator:
def __init__(self, function):
self.function = function
def __call__(self, *args, **kwargs):
# We can add some code
# before function call
self.function(*args, **kwargs)
# We can also add some code
# after function call.
# adding class decorator to the function
@MyDecorator
def function(name, message ='Hello'):
print("{}, {}".format(message, name))
用类定义装饰器,然后装饰一个类
先定义装饰器:
class MyClassDecorator(object):
_instances = dict()
def __init__(self, name):
pass
def __call__(self, cls):
class WrappedClass(cls):
def say_hello(self):
print(f'Hello: {self.username}')
return WrappedClass
该装饰器给被装饰的类上添加了一个方法,名称为say_hello()。使用如下:
@MyClassDecorator('test')
class MyClass():
def __init__(self, username):
self.username = username
然后:
def test_decoratorforclass():
obj = MyClass('user1')
obj.say_hello()
打印出: Hello: user1
小结
学习类装饰,对Python的内部机制会有更多的了解。如__init__, call, __new__等内置方法。
来源:https://blog.csdn.net/jgku/article/details/128020535
0
投稿
猜你喜欢
- 一、程序导出word文档的方法将web/html内容导出为world文档,再java中有很多解决方案,比如使用Jacob、Apache PO
- 有使用过VS2005开发工具的朋友或者其他语句如js中都有Try catch 语句块,那么在mysql中是否能有SQLserver的@@er
- 作为课代表,经常要做的两件事是:帮忙发作业和帮忙收作业,而且很多时候是通过邮件来完成的,如果手动一封一封的收取和发送就很浪费时间——人生苦短
- 一定要注重代码规范,按照平时的代码管理,可以将Python代码规范检测分为两种:静态本地检测:可以借助静态检查工具,比如:Flake8,Py
- 每个PHP脚本都限制了执行时间,所以我们需要通过 set_time_limit 来设置一个脚本的执行时间为无限长;然后使用 flush()
- 先说迭代器,对于string、list、dict、tuple等这类容器对象,使用for循环遍历是很方便的。在后台for语句对容器对象调用it
- cupy我觉得可以理解为cuda for numpy,安装方式pip install cupy,假设import numpy as npim
- 简单邮件传输协议(SMTP)是一种协议,用于处理在电子邮件服务器之间发送电子邮件和路由电子邮件。Python提供了smtplib模块,该模块
- 1、实现 __getitem__(self)class Library(object): def __init__(self):
- 复习回顾:Python 对于时间日期操作提供了很多方法,我们前面已经学习了2个模块:基于Unix 时间戳范围限制在1970~2038年的时间
- Oracle 背景资料 在介绍 Oracle9i 之前我们先介绍一些关于Oracle 公司的资料,让各位朋友更多了解 Oracle。 197
- 好多次在不同场合,都听到有同行提到“做产品比做设计更有前途”,或者“别做设计了,做产品吧”类似的观点。我不认为它们之间有什么可比性,因为这么
- 这个是我在蓝色看到的,楼主想实现图片按比例缩放的功能(缩略图),把图片固定在一定的宽高范围内,不会变形,失真。例如:缩略图的框是94px*9
- http://validator.w3.org/#validate_by_upload 在线校验网址点浏览,上次找到自己做的页面
- 如今WEB的安全问题影响着整个安全界,SQL注入,跨站脚本攻击等攻击受到了关注。 网络安全问题日益变的更加重要,国内依然有很多主机受到此类安
- 关于 PHP 的文件操作,我们也将是通过一系列的文章来进行学习。今天我们先学习的是一个很少人使用过,甚至很多人根本不知道的扩展,它与我们日常
- 登录与注册两个按钮似乎天生就应该是排在一起的,就像很多地方的“确定”与“取消”一样,甚至排在一起的意义远远强于后者。于是长期以来,用户们也形
- Selenium简介Selenium是一个用于测试网站的自动化测试工具,支持各种浏览器包括Chrome、Firefox、Safari等主流界
- PHP的类是单一继承模式,也就是每个类只能继承一个父类(基类)。但有时需要引入更多通用(共用)的方法,同时这些方法又不适合集成到基类。那么这
- 前言上一篇介绍了服务端流式RPC,客户端发送请求到服务器,拿到一个流去读取返回的消息序列。 客户端读取返回的流的数据。本篇将介绍客户端流式R