Python装饰器

函数装饰器(decorator)用于在源码中“标记”函数,在不修改原函数的前提下,以某种方式增强函数的行为。

Python的装饰器基于闭包实现。闭包是指函数中再嵌套一个函数,并且引用外部函数的变量。

1
2
3
4
5
6
7
8
# 在该示例中,outer函数内又定义了一个inner函数,并且inner函数又引用了外部函数outer的变量x,形成一个闭包
def outer(x):
    def inner(y):
        return x + y
    return inner

# outer(1)(2)中,第一个括号传进去的值返回inner函数,其实就是返回1 + y,所以再传第二个参数进去,就可以得到返回值,1 + 2
print(outer(1)(2))
1
3

装饰器本质上就是一种闭包。我们可以用如下的方式实现一个装饰器:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
import time

# 该装饰器可以对修饰的函数进行计时
def clock(func):
    def clocked(*args): # 
        t0 = time.perf_counter()
        result = func(*args) # 
        elapsed = time.perf_counter() - t0
        name = func.__name__
        arg_str = ', '.join(repr(arg) for arg in args)
        print('[%0.8fs] %s(%s) -> %r' % (elapsed, name, arg_str, result))
        return result
    return clocked

@clock
def factorial(n):
    return 1 if n < 2 else n*factorial(n-1)

factorial(5)

# 这一写法实际上等效于以下写法,可以看到装饰器就是一种更加简洁的闭包写法:
# clock(factorial)(5)
1
2
3
4
5
[0.00000050s] factorial(1) -> 1
[0.00004400s] factorial(2) -> 2
[0.00008740s] factorial(3) -> 6
[0.00012330s] factorial(4) -> 24
[0.00013580s] factorial(5) -> 120

内置装饰器

实际上,我们自定义Python装饰器的情况还是比较少的。我们大多数时候使用Python所内置的一些装饰器来管理函数。如@contextmanager, @staticmethod, @classmethod, @abstractmethod, @property等。这些装饰器的用法将在后续教程中陆续介绍。

Licensed under CC BY-NC-SA 4.0
Built with Hugo
Theme Stack designed by Jimmy
visitors: total visits: time(s) reads: time(s)