Python 装饰器(Decorator)

1.什么是装饰器

python中的装饰器是增强函数或类的功能的一个函数 / 类,是闭包(closure)的一种应用。
通过使用装饰器,我们可以在不修改原函数代码的前提下拓展函数的功能。

例如下面的代码中,func()函数的上方的@demo就是装饰器。

@demo
def func():
    print('This is func')

2.为什么要使用装饰器

假设有这么一个函数

import time 

def func1():
    print("This is func1")
    time.sleep(2) 

现在,我们需要计算并输出这个函数执行所花费的时间。
使用传统方法改造函数,可能会是下面这个样子:

import time

def func1():
    start_time = time.time()    
    print("This is func1")
    time.sleep(2) 
    exec_time = time.time() - start_time    
    print('func1 :', exec_time) 

func1()
-----------------------------
>>>This is func1
>>>func1 : 2.002577066421509

如果又有另外一个函数func2()也同样需要计算时间,那就需要也对func2()进行同样的改造。

import time

def func1():
    start_time = time.time()    
    print("This is func1")
    time.sleep(2) 
    exec_time = time.time() - start_time    
    print('func1 :', exec_time) 

def func2():
    start_time = time.time()    
    print("This is func2")
    time.sleep(5) 
    exec_time = time.time() - start_time    
    print('func2 :', exec_time) 

func1()
func2()
-----------------------------
>>>This is func1
>>>func1 : 2.002577066421509
>>>This is func2
>>>func2 : 5.000931262969971

像这样如果有n个函数需要改造,那就需要把同样的逻辑写n遍。这会带来几个问题:

  • 工作量很大
  • 如果计算时间的代码有改动,需要修改每个函数
  • 降低了代码的可读性,不够简洁优雅

所以,我们需要更好的解决方法。

3.如何使用装饰器

回到上面的例子,这次我们使用闭包来实现功能。

import time

def time_calc(func):
    def wrapper():        
        start_time = time.time()        
        f = func()        
        exec_time = time.time() - start_time 
        print('{} :{}'.format(func.__name__, exec_time))        
        return f    
    return wrapper   

def func1():
    print("This is func1")
    time.sleep(2) 

def func2():
    print("This is func2")
    time.sleep(5) 

decorated1 = time_calc(func1)
decorated1()

decorated2 = time_calc(func2)
decorated2()
-----------------------------
>>>This is func1
>>>func1 :2.0153732299804688
>>>This is func2
>>>func2 :5.0016069412231445

可以看到在time_calc()函数内,又定义了一个wrapper()函数,并且wrapper()函数又引用了外部函数time_calc()的变量func,形成了一个闭包。
使用时,我们首先将真正要执行的函数(例子中的func1()或者func2())作为参数传递给time_calc()函数。time_calc()函数在其前后增加额外的逻辑后再将包装好的函数返回给我们。我们通过执行包装好的函数实现所有功能。

相比直接修改函数,显然使用闭包的方案代码复用率更高,也更易于维护。
但是这也带来了新的问题:执行函数前必须先对函数进行包装,然后执行包装后的函数。

decorated1 = time_calc(func1)
decorated1()

为了解决这一问题,Python 2.4通过在函数定义前添加一个装饰器名和@符号,来实现对函数的自动包装。
下面是装饰器方案的代码:

import time

def time_calc(func):
    def wrapper():        
        start_time = time.time()        
        f = func()        
        exec_time = time.time() - start_time 
        print('{} :{}'.format(func.__name__, exec_time))        
        return f    
    return wrapper   

@time_calc
def func1():
    print("This is func1")
    time.sleep(2) 

@time_calc
def func2():
    print("This is func2")
    time.sleep(5) 

func1()
func2()
-----------------------------
>>>This is func1
>>>func1 :2.011098861694336
>>>This is func2
>>>func2 :5.007617473602295

可以看到,在函数定义的上方添加@time_calc后,无需手动包装,直接执行原函数就可以达到一样的效果。
这是因为@time_calc等同于下面的代码:

func1 = time_calc(func1)

装饰器还可以嵌套

@deco1
@deco2
@deco3
def func()
    pass

装饰器嵌套时执行的顺序为由下至上,上面的代码等同于:

func = deco1(deco2(deco3(func)))

4.带参数的函数与带参数的装饰器

很多时候(也许是绝大多数时候)我们执行的函数是需要传入参数的。例如:

def func_add(x, y):
    print('x + y = ', x+y)
    time.sleep(1)

这时候,我们需要修改装饰器使其可以接收函数的参数。

import time

def time_calc(func):
    def wrapper(x, y):        
        start_time = time.time()        
        f = func(x, y)        
        exec_time = time.time() - start_time 
        print('{} :{}'.format(func.__name__, exec_time))        
        return f    
    return wrapper   

@time_calc
def func_add(x, y):
    print('x + y = ', x+y)
    time.sleep(1)
    
func_add(1, 2)
-----------------------------
>>>x + y =  3
>>>func_add :1.0004308223724365

由于一个装饰器往往要对应许多个不同的函数,而这些函数又很有可能拥有不同的参数数量。为了提高泛用性,一般使用*args**kwargs来取代具体的参数和键值对。

import time

def time_calc(func):
    def wrapper(*args, **kwargs):        
        start_time = time.time()        
        f = func(*args, **kwargs)        
        exec_time = time.time() - start_time 
        print('{} :{}'.format(func.__name__, exec_time))        
        return f    
    return wrapper   

@time_calc
def func_add(x, y):
    print('x + y = ', x+y)
    time.sleep(1)

@time_calc
def func_add_three(x, y, z):
    print('x + y + z = ', x+y+z)
    time.sleep(2)
    
func_add(1, 2)
func_add_three(4, 5, 6)
-----------------------------
>>>x + y =  3
>>>func_add :1.0120115280151367
>>>x + y + z =  15
>>>func_add_three :2.0111453533172607

另外,我们也可以对装饰器本身传递参数,只需在原有的基础上再嵌套一层即可。
例如下面代码中的装饰器能够接收一个level参数并将参数的值输出到打印结果中。

import time

def time_calc(level):
    def outwrapper(func):
        def wrapper(*args, **kwargs):        
            start_time = time.time()        
            f = func(*args, **kwargs)        
            exec_time = time.time() - start_time
            print('{} {} :{}'.format(level, func.__name__, exec_time))        
            return f   
        return wrapper    
    return outwrapper   

@time_calc(level="INFO")
def func_add(x, y):
    print('x + y = ', x+y)
    time.sleep(1)

@time_calc(level="DEBUG")
def func_add_three(x, y, z):
    print('x + y + z = ', x+y+z)
    time.sleep(2)
    
func_add(1, 2)
func_add_three(4, 5, 6)
-----------------------------
>>>x + y =  3
>>>INFO func_add :1.015167236328125
>>>x + y + z =  15
>>>DEBUG func_add_three :2.0142157077789307

5.类装饰器

除了函数,还可以使用类作为装饰器。

import time

class TimeCalculator:
    def __init__(self, func):
       self.func = func

    def __call__(self, *args, **kwargs):
        start_time = time.time()        
        f = self.func(*args, **kwargs)        
        exec_time = time.time() - start_time
        print('{} :{}'.format(self.func.__name__, exec_time))        
        return f  

@TimeCalculator
def func_add(x, y):
    print('x + y = ', x+y)
    time.sleep(1)

@TimeCalculator
def func_add_three(x, y, z):
    print('x + y + z = ', x+y+z)
    time.sleep(2)
    
func_add(1, 2)
func_add_three(4, 5, 6)
-----------------------------
>>>x + y =  3
>>>func_add :1.0122718811035156
>>>x + y + z =  15
>>>func_add_three :2.006585121154785

用法与函数装饰器并没有太大区别,实质是使用了类方法中的__call__()方法来实现类的直接调用。

当然,类装饰器也是可以带参数的:

import time

class TimeCalculator:
    def __init__(self, level):
       self.level = level

    def __call__(self, func):
        def wrapper(*args, **kwargs):
            start_time = time.time()        
            f = func(*args, **kwargs)        
            exec_time = time.time() - start_time
            print('{} {} :{}'.format(self.level, func.__name__, exec_time))        
            return f
        return wrapper

@TimeCalculator(level='INFO')
def func_add(x, y):
    print('x + y = ', x+y)
    time.sleep(1)

@TimeCalculator(level='DEBUG')
def func_add_three(x, y, z):
    print('x + y + z = ', x+y+z)
    time.sleep(2)
    
func_add(1, 2)
func_add_three(4, 5, 6)

此时__init__()方法负责接收传递给类装饰器的参数,__call__()方法负责接收要执行的函数。
__call__()方法里面增加了一层嵌套用来接收要执行的函数的参数。

相比函数装饰器,类装饰器具有一定的优势。
比如可以在类中定义多个函数以对应不同的用途,还可以利用类的成员变量来储存信息等。
另外由于是类,当然也可以被继承。
所以如果需要实现的功能比较复杂,也许使用类装饰器是一个更好的选择。

下面是一个缓存装饰器的例子。

  • 使用函数名和参数生成key,保存对应的函数执行结果到成员变量data里。
  • 如果拥有相同参数的函数已经执行过了,则直接返回成员变量data里的结果,并打印“cache result”。
  • 如果没有执行过,则执行函数返回结果,并打印“calculation result”。
class Cacher:
    def __init__(self, func):
        self.func = func
        self.data = {}

    def __call__(self, *args, **kwargs):
        key = f'{self.func.__name__}-{str(args)}-{str(kwargs)})'
        if key in self.data:
            result = self.data.get(key)
            print('cache result')
        else:
            result = self.func(*args, **kwargs)
            self.data[key] = result
            print('calculation result')
        return result

@Cacher
def func_add(x, y):
    return x + y

print(func_add(1, 2))
print(func_add(3, 4))
print(func_add(1, 2))
-----------------------------
>>>calculation result
>>>3
>>>calculation result
>>>7
>>>cache result
>>>3

此外,类的实例对象也可以作为装饰器使用,在这里就不举例说明了。

6.python的几种内置装饰器

python已经为我们内置了几种装饰器,可以直接使用。

@property

property装饰器可以把一个实例方法变成其同名属性,以支持实例访问,它返回的是一个property属性。
例如以下代码中给方法x()添加@property装饰器后就可以像访问数据属性一样去访问它。

class C:
    def __init__(self, x=None):
        self._x = x

    @property
    def x(self):
        return self._x

c = C(100)
print(c.x)
-----------------------------
>>>100

一个property对象还具有setter、deleter和getter装饰器。

  • setter用于设置属性值
  • deleter用于删除属性值
  • getter用于获取属性值(但是因为可以直接通过property获取属性信息所以一般不太有人用它)

下面的代码在初始化后分别执行了对属性_x的赋值和删除。执行最后一行print(c.x)时,由于属性_x已经被删除,所以程序输出了出错信息。

class C:
    def __init__(self, x=None):
        self._x = x

    @property
    def x(self):
        return self._x

    @x.setter
    def x(self, value):
        print('set _x to', value)
        self._x = value

    @x.deleter
    def x(self):
        print('delete _x')
        del self._x

c = C()
print(c.x)
c.x = 100
print(c.x)
del c.x
print(c.x)
-----------------------------
>>>None
>>>set _x to 100
>>>100
>>>delete _x
>>>Traceback (most recent call last):
   ...
   AttributeError: 'C' object has no attribute '_x'
@classmethod

被classmethod装饰器后方法可以直接通过类名来直接调用,而无需建立实例。
与实例方法需要通过self参数隐式的传递当前类对象的实例相似,@classmethod修饰的方法也需要通过第一参数cls传递当前类对象。
由于是类方法,只能访问类属性,不能访问实例方法和实例属性。

class C:
    classParam = 'classparam'    
    def __init__(self, x=None):
        self._x = x

    def printInstanceParam (self):
        print(self._x)

    @classmethod  
    def printClassParam(cls):
        print(cls.classParam)

C.printClassParam()
C.printInstanceParam()
-----------------------------
>>>classparam
>>>Traceback (most recent call last):
   ...
   TypeError: printInstanceParam() missing 1 required positional argument: 'self'
@staticmethod

staticmethod装饰器可以将类方法改变为静态方法。
静态方法不需要传递隐性的第一参数,可以直接通过类进行调用,也可以通过实例进行调用。
静态方法的本质类型就是一个函数,只是寄存在一个类名下。

class C(object):
    @staticmethod
    def func():
        print('run func');
 
C.func(); 
cobj = C()
cobj.func()
-----------------------------
>>>run func
>>>run func
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,616评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,020评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,078评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,040评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,154评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,265评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,298评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,072评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,491评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,795评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,970评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,654评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,272评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,985评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,815评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,852评论 2 351

推荐阅读更多精彩内容