详解Python装饰器的四种定义形式

作者:北极象 时间:2022-06-20 13:35:04 

前言

装饰器(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

标签:Python,装饰器
0
投稿

猜你喜欢

  • Mysql实现简易版搜索引擎的示例代码

    2024-01-28 10:42:32
  • 在python plt图表中文字大小调节的方法

    2021-04-21 04:40:28
  • go语言代码生成器code generator使用示例介绍

    2024-05-21 10:19:29
  • Python代码实现删除一个list里面重复元素的方法

    2022-06-02 15:31:02
  • Go语言文件读取的一些总结

    2024-04-27 15:30:45
  • 一条SQL语句修改多表多字段的信息的具体实现

    2024-01-18 13:22:56
  • Oracle生成单据编号存储过程的实例代码

    2024-01-23 19:10:19
  • pycharm2022.2远程连接服务器调试代码实现

    2022-11-17 12:12:48
  • MySql节点管理安装步骤

    2010-10-14 14:13:00
  • asp使用Application来统计在线人数方法

    2007-08-13 12:43:00
  • 超越MYSQL,ACCESS复合承载

    2008-12-09 13:31:00
  • Python实现一键抠图的示例代码

    2022-01-31 10:15:49
  • Go Web 编程中的模板库应用指南(超详细)

    2024-02-05 20:50:55
  • Golang源码分析之golang/sync之singleflight

    2024-04-25 15:07:26
  • 在Python中操作时间之tzset()方法的使用教程

    2022-10-28 22:22:01
  • Django静态文件加载失败解决方案

    2021-05-25 19:40:47
  • Mysql覆盖索引详解

    2024-01-14 06:54:29
  • asp 获取url函数小结

    2011-03-17 10:38:00
  • python 如何在list中找Topk的数值和索引

    2022-01-20 10:28:27
  • python实现输出一个序列的所有子序列示例

    2022-04-13 18:34:45
  • asp之家 网络编程 m.aspxhome.com