【第十九天】带参数的装饰器与装饰类

2.带参装饰器

在上面的装饰器调用中,比如@decorator_demo
该装饰器默认它后面的函数是唯一的参数
装饰器的语法允许我们调用decorator时,提供其他参数
比如@decorator(a),这样,就为装饰器的编写和使用提供了更大的灵活性

#带参装饰器

def pre_str(pre=""):
    def decorator(old_function):
        def new_function(a,b):
            print(pre + "input",a,b)
            return old_function(a,b)
        return new_function
    return decorator
    
#装饰square_sum()
@pre_str("^_^")
def square_sum(a,b):
    return a**2 + b**2   
    
#装饰square_diff()
@pre_str("T_T")
def square_diff(a,b):
    return a**2 - b**2

if __name__ == '__main__':
    print(square_sum(3,4))
    print(square_diff(3,4))
^_^input 3 4
25
T_Tinput 3 4
-7

上面的pre_str是一个带参装饰器,它实际上是对原有装饰器的一个函数封装
并返回一个装饰器,我们可以将它理解为一个含有环境参量的闭包
当我们使用@pre_str("_")调用的时候,py能够发现这一层的封装
并把参数传递到装饰器的环境中,该调用相当于:

square_sum = pre_str("^_^")(square_sum)

根据参数不同,带参装饰器会对函数进行不同的加工,进一步提高了装饰器的适用范围
还是以网站的用户验证为例子,装饰器负责验证的功能,装饰了处理HTTP请求的函数
可能有的关键HTTP请求需要管理员权限,有的只需要普通用户权限
因此,我们可以把“管理员”和“用户”作为参数,传递给验证装饰器
对于那些负责关键HTTP请求的函数,我们可以把“管理员”参数传给装饰器
对于负责普通HTTP请求的函数,我们可以把“用户”参数传给它们的装饰器
这样,同一个装饰器就可以满足不同的需求了

3.装饰类

在上面的例子中,装饰器接收一个函数,并返回一个函数
从而起到加工函数的效果,装饰器还拓展到了类
一个装饰器可以接收一个类,并返回一个类,从而起到加工类的效果

def decorator_class(SomeClass):
    class NewClass(object):
        def __init__(self,age):
            self.total_display = 0
            self.wrapped = SomeClass(age)
        def display(self):
            self.total_display += 1
            print('total display',self.total_display)
            self.wrapped.display()
    return NewClass

@decorator_class
class Bird:
    def __init__(self,age):
        self.age = age
    def display(self):
        print('my age is',self.age)
        
if __name__ == '__main__':
    eagle_lord = Bird(5)
    for i in range(3):
        eagle_lord.display()
total display 1
my age is 5
total display 2
my age is 5
total display 3
my age is 5

在装饰器decorator_class中,我们返回了一个新类NewClass
在新类的构造器中,我们用一个属性self.wrapped记录了原来类生成的对象
并附加了新的属性total_display,用于记录调用display()的次数
我们也同时更改了display方法,通过装饰
我们的Bird类可以调用display()的次数
无论是装饰函数,还是装饰类,装饰器的核心作用都是名称绑定
虽然装饰器出现较晚,但在各个py项目中的使用却很广泛
即便不需要自定义装饰器,你也很有可能会在自己的项目中调用其他库中的装饰器
因此,py程序员需要掌握这一语法

你可能感兴趣的:(【第十九天】带参数的装饰器与装饰类)