Python装饰器本质是闭包与高阶函数的自然产物,是接收函数并返回新函数的“函数工厂”;@语法糖等价于手动赋值,带参装饰器需三层嵌套,须用functools.wraps保留元信息,类装饰器适合管理状态但需实现__call__。
Python 装饰器不是语法糖的“魔法”,而是函数式编程中 闭包 + 高阶函数 的自然产物;只要理解 def 语句执行时创建函数对象、函数可赋值/传参/返回的本质,装饰器就不再黑盒。
所谓 @log_time,等价于:
def my_func():
pass
my_func = log_time(my_func)这意味着:
log_time 必须是可调用对象,且返回值也必须是可调用对象(通常是函数)__name__、__doc__ 等元信息默认会丢失——因为实际运行的是包装后的函数log_time 接收参数(如 @log_time(level='DEBUG')),它必须再套一层函数:外层接收参数,返回内层装饰器常见错误是写成两层,导致解释器报错 TypeError: 'function' object is not callable 或直接跳过装饰逻辑。正确结构:
def retry(max_attempts=3, delay=1):
def decorator(func): # 第二层:真正的装饰器
def wrapper(*args, **kwargs):
for i in range(max_attempts):
try:
return func(*args, **kwargs)
except Exception:
if i == max_attempts - 1:
raise
time.sleep(delay)
return wrapper
return decorator # 第一层返回装饰器关键点:
retry() 接收配置参数,返回第二层 decorator
decorator 接收被装饰函数,返回第三层 wrapper
略任意一层,否则 @retry() 无法解析为合法装饰器调用functools.wraps 修复元信息丢失问题不加 @wraps(func),help(my_decorated_func) 显示的是 wrapper 的签名和文档,而非原函数。修复只需一行:
from functools import wrapsdef log_call(func): @wraps(func) # ← 这行必须有,且必须在 wrapper 定义后立即使用 def wrapper(*args, *kwargs): print(f"Calling {func.name}") return func(args, **kwargs) return wrapper
它等价于手动复制:wrapper.__name__ = func.__name__、wrapper.__doc__ = func.__doc__ 等。漏掉这行,调试和 IDE 提示会严重失真。
当需要在多次调用间共享状态(如计数、缓存、连接池),类装饰器更直观:
class CountCalls:
def __init__(self, func):
self.func = func
self.count = 0
wraps(func)(self) # 同样要保留原函数元信息
def __call__(self, *args, **kwargs):
self.count += 1
return self.func(*args, **kwargs)注意:
__call__ 才能被调用;否则 @CountCalls 会报 TypeError: 'CountCalls' object is not callable
wraps(func)(self) 是把原函数元信息绑定到实例上,不是绑定到 __call__ 方法@CountCalls(threshold=10)),需额外加一层工厂类或改用函数式三层结构真正卡住人的从来不是语法,而是没想清楚「谁在什么时候被调用」:装饰器定义时执行?装饰时执行?还是被装饰函数调用时执行?把这三阶段拆开验证,比死记模板有效得多。