Python装饰器本质是接收函数并返回新函数的“增强器”,通过包装动态添加行为;其执行逻辑为@decorator等价于func = decorator(func),需确保装饰器及返回值均可调用。
Python装饰器的本质是函数的“增强器”——它不修改原函数代码,而是通过包装(wrap)方式动态添加新行为。理解这一点,就抓住了装饰器的核心:接收函数、返回函数、运行时介入。
装饰器实际是“函数调用 + 返回可调用对象”的组合。当写 @log_time 时,Python 会自动执行 func = log_time(func)。因此,装饰器本身必须是可调用对象(通常是函数),且返回值也必须是可调用对象(常为内部闭包函数)。
wrapper),完成前置/后置逻辑,并调用原函数functools.wraps(func) 可保留原函数的 __name__、__doc__ 等元信息,避免调试时混淆带参数的装饰器其实是“三层函数”结构:参数接收层 → 装饰器工厂层 → 包装执行层。例如 @retry(max_times=3)
并非直接作用于函数,而是先调用 retry(max_times=3) 得到真正的装饰器,再用它去装饰目标函数。
max_times),返回第二层函数wrapper
wrapper 接收原函数的参数,控制逻辑并调用原函数return,导致返回 None,引发 TypeError: 'NoneType' object is not callable
真实项目里,装饰器不是炫技工具,而是解决重复横切关注点的利器。下面几个案例覆盖大多数需求:
functools.lru_cache 是内置轻量方案;自定义可结合 Redis 控制过期与键生成逻辑装饰器看着简洁,但稍不注意就会引发隐蔽问题:
*args, **kwargs 在 wrapper 中透传,别硬写参数名self 或 cls 是第一个参数,wrapper 必须兼容@auth @log @cache)?执行顺序是自下而上,即 @cache 最先被应用,@auth 最后生效wrapper 内部处理,而不是在装饰器定义时就执行