装饰器基础知识
装饰器是可调用对象,其参数是另一个函数(被装饰的函数),装饰器可能会处理被装饰的函数,然后把它返回,或者将其替换成另一个函数或者可调用对象,例如:
def deco(func): # 1. 接收被装饰的函数作为参数
def inner():
print('running inner()...')
return inner # 2. 返回 inner 函数,原函数被替换
# 3. 装饰在函数上
@deco
def target():
print('running target() ...')
# 3.1 普通写法
target_ = deco(target)
# 4. 调用被装饰的 target 实际上会运行 inner
target() # running inner()...
# 4.1 这两种写法效果都一样
target_() # running inner()...
# 审查对象,发现 target 和 target_ 都是 inner 的引用
print(target) # <function deco.<locals>.inner at 0x106aad158>
print(target_) # <function deco.<locals>.inner at 0x106aad1e0>
严格来讲,装饰器只是语法糖,它可以像常规的可调用对象那样调用,其参数是另外一个函数。
装饰器有两大特性:1. 能把被装饰的函数替换成其他函数; 2. 它在被装饰的函数定义时立即执行;
Python 何时执行装饰器
装饰器的一个特性是:它在被装饰的函数定义时立即执行,通常是在导入时执行(通常情况下,装饰器是在另外一个模块中);
假使装饰器和被装饰的函数不在同一个模块:
from decorator import deco
print('###### the first breakpoint ######')
# 用装饰器装饰函数
@deco
def target():
print('running target() ...')
if __name__ == '__main__':
print('start main ------')
# 调用被装饰的函数
target()
print('main end ------')
# 控制台输出:
# ##### the first breakpoint ######
# running deco() ...
# start main ------
# running wrapper() ...
# main end ------
综上,可以看出装饰器是在被装饰的函数定义后立即执行。