• 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
    
    • 1
    • 2
    • 3
    • 4

    然后作用在一个函数上:

    @decorate
    def add(a, b):
    	return a + b
    
    • 1
    • 2
    • 3

    测试一下效果:

    def test_decorate():
    	sum = add(3, 5)
    	assert sum == 16
    
    • 1
    • 2
    • 3

    用函数装饰一个类

    这里通过装饰器实现单例模式:

    def singleton(cls):
        instances = {}
        def wrapper(*args, **kwargs):
            if cls not in instances:
              instances[cls] = cls(*args, **kwargs)
            return instances[cls]
        return wrapper
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    使用该装饰器:

    @singleton
    class MyClass:
        def method(self):
            pass
    
    • 1
    • 2
    • 3
    • 4

    于是,当你定义多个对象时,返回的是同一实例:

    obj = MyClass()  # creates a new instance
    obj2 = MyClass()  # returns the same instance
    obj3 = MyClass()  # returns the same instance
    ...
    
    • 1
    • 2
    • 3
    • 4

    用类定义装饰器,然后装饰一个函数

    先采用类定义一个装饰器:

    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
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10

    再作用在一个函数上:

    @Star(5)
    def add(a, b):
        return a + b
    
    • 1
    • 2
    • 3

    主要是在类中实现__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))
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19

    用类定义装饰器,然后装饰一个类

    先定义装饰器:

    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
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11

    该装饰器给被装饰的类上添加了一个方法,名称为say_hello()。使用如下:

    @MyClassDecorator('test')
    class MyClass():
    	def __init__(self, username):
    		self.username = username
    
    • 1
    • 2
    • 3
    • 4

    然后:

    def test_decoratorforclass():
    	obj = MyClass('user1')
    	obj.say_hello()
    
    • 1
    • 2
    • 3

    打印出: Hello: user1

    小结

    学习类装饰,对Python的内部机制会有更多的了解。如__init__, call, __new__等内置方法。

  • 相关阅读:
    并发编程之ForkJoin框架
    排查Linux服务器是否被入侵步骤
    大数据之Hive(二)
    主流定时任务解决方案全横评
    C++ 模板 (一)
    279. 完全平方数
    Go语言笔记-基础篇
    实现支持 MJPEG 的播放器
    音频学习笔记之音频播放
    Michael.W基于Foundry精读Openzeppelin第35期——Ownable.sol
  • 原文地址:https://blog.csdn.net/jgku/article/details/128020535