0

0

Python 装饰器叠加时的执行顺序

冷漠man

冷漠man

发布时间:2026-01-28 19:27:10

|

270人浏览过

|

来源于php中文网

原创

装饰器叠加时执行顺序与书写顺序相反:最下方的装饰器最先执行,最上方的最后执行,即f = a(b(f));带参装饰器需先求值得到实际装饰器再嵌套;每层应使用@wraps保持元信息,调试宜分阶段加日志。

python 装饰器叠加时的执行顺序

装饰器叠加时,@decorator 的书写顺序和实际执行顺序相反

写在最上面的装饰器,最先被应用(即最晚执行),而写在最下面的,最先执行。这不是 Python 的“反直觉”,而是语法糖展开后的自然结果:@a\n@b\ndef f(): ... 等价于 f = a(b(f)) —— 从内到外套用函数。

常见错误现象:以为 @log 写在最上就最先打印日志,结果发现它反而在最里层执行,甚至没看到日志就抛出了异常。

  • 执行顺序是:最下层装饰器 → 中间层 → 最上层(即包裹顺序的逆序)
  • 返回值传递方向是:原函数 → 最下层装饰器 → … → 最上层装饰器 → 调用者
  • 若某层装饰器没调用 func(*args, **kwargs),后续所有上层逻辑都会被跳过

带参数的装饰器叠加时,先求值再套用

@retry(max_attempts=3) 这类装饰器本身不是装饰器,而是「返回装饰器的函数」。叠加时,Python 先执行 retry(max_attempts=3) 得到真正的装饰器,再参与 a(b(f)) 式嵌套。

容易踩的坑:在参数函数里做耗时操作(比如读配置、连数据库),会导致每次导入模块时就执行,而不是等到函数被调用。

立即学习Python免费学习笔记(深入)”;

  • 确保参数化装饰器的外层函数(如 retry())只做必要初始化,不触发业务逻辑
  • 内层闭包(真正返回的装饰器)才应包含运行时逻辑(如重试判断)
  • 调试时可打印每层装饰器的定义时机,区分「定义期」和「调用期」

装饰器中 functools.wraps 只影响最外层函数元信息

叠加三层装饰器后,若只有最外层用了 @wraps(func),那么 help()__name__ 等只还原到被装饰的原函数;中间层若没用 wraps,其包装函数的 __doc____module__ 会暴露出来,导致调试困难。

PLC编程入门基础知识 中文doc版
PLC编程入门基础知识 中文doc版

可编程序控制器,英文称Programmable Controller,简称PC。但由于PC容易和个人计算机(Personal Computer)混淆,故人们仍习惯地用PLC作为可编程序控制器的缩写。它是一个以微处理器为核心的数字运算操作的电子系统装置,专为在工业现场应用而设计,它采用可编程序的存储器,用以在其内部存储执行逻辑运算、顺序控制、定时/计数和算术运算等操作指令,并通过数字式或模拟式的输入、输出接口,控制各种类型的机械或生产过程。本平台提供PLC编程入门基础知识下载,需要的朋友们下载看看吧!

下载

典型表现:inspect.signature(f) 拿不到原函数参数,或 IDE 提示参数为 *args, **kwargs

  • 每一层装饰器内部,只要返回了新函数,就应使用 @wraps(original_func)
  • 如果某层装饰器做了参数改写(如注入 db_session),需配合 signature 手动更新,仅靠 wraps 不够
  • help(f) 快速验证:正确叠加后应显示原函数的文档和签名

调试装饰器执行流的实用方法

不要靠猜,直接在各层装饰器的入口和出口加日志(注意别污染生产环境)。关键是区分「装饰时」和「调用时」两个阶段。

一个可靠示例:

def trace(name):
    def decorator(func):
        print(f"[装饰阶段] {name} 正在包装 {func.__name__}")
        @wraps(func)
        def wrapper(*args, **kwargs):
            print(f"[调用阶段] → 进入 {name}.{func.__name__}")
            result = func(*args, **kwargs)
            print(f"[调用阶段] ← 退出 {name}.{func.__name__}")
            return result
        return wrapper
    return decorator

@trace("outer")
@trace("inner")
def say():
    return "done"

运行后输出顺序清晰可见:先打印两行「装饰阶段」,调用 say() 时才按 inner → outer 执行「调用阶段」。

复杂点在于,有些装饰器(如 @lru_cache)在装饰阶段就做了不可逆初始化,叠加时顺序错可能导致缓存键计算逻辑被覆盖——这种必须严格按语义分层,不能只看执行顺序。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
java值传递和引用传递有什么区别
java值传递和引用传递有什么区别

java值传递和引用传递的区别:1、基本数据类型的传递;2、对象的传递;3、修改引用指向的情况。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

108

2024.02.23

go语言闭包相关教程大全
go语言闭包相关教程大全

本专题整合了go语言闭包相关数据,阅读专题下面的文章了解更多相关内容。

137

2025.07.29

数据库三范式
数据库三范式

数据库三范式是一种设计规范,用于规范化关系型数据库中的数据结构,它通过消除冗余数据、提高数据库性能和数据一致性,提供了一种有效的数据库设计方法。本专题提供数据库三范式相关的文章、下载和课程。

358

2023.06.29

如何删除数据库
如何删除数据库

删除数据库是指在MySQL中完全移除一个数据库及其所包含的所有数据和结构,作用包括:1、释放存储空间;2、确保数据的安全性;3、提高数据库的整体性能,加速查询和操作的执行速度。尽管删除数据库具有一些好处,但在执行任何删除操作之前,务必谨慎操作,并备份重要的数据。删除数据库将永久性地删除所有相关数据和结构,无法回滚。

2082

2023.08.14

vb怎么连接数据库
vb怎么连接数据库

在VB中,连接数据库通常使用ADO(ActiveX 数据对象)或 DAO(Data Access Objects)这两个技术来实现:1、引入ADO库;2、创建ADO连接对象;3、配置连接字符串;4、打开连接;5、执行SQL语句;6、处理查询结果;7、关闭连接即可。

349

2023.08.31

MySQL恢复数据库
MySQL恢复数据库

MySQL恢复数据库的方法有使用物理备份恢复、使用逻辑备份恢复、使用二进制日志恢复和使用数据库复制进行恢复等。本专题为大家提供MySQL数据库相关的文章、下载、课程内容,供大家免费下载体验。

256

2023.09.05

vb中怎么连接access数据库
vb中怎么连接access数据库

vb中连接access数据库的步骤包括引用必要的命名空间、创建连接字符串、创建连接对象、打开连接、执行SQL语句和关闭连接。本专题为大家提供连接access数据库相关的文章、下载、课程内容,供大家免费下载体验。

326

2023.10.09

数据库对象名无效怎么解决
数据库对象名无效怎么解决

数据库对象名无效解决办法:1、检查使用的对象名是否正确,确保没有拼写错误;2、检查数据库中是否已存在具有相同名称的对象,如果是,请更改对象名为一个不同的名称,然后重新创建;3、确保在连接数据库时使用了正确的用户名、密码和数据库名称;4、尝试重启数据库服务,然后再次尝试创建或使用对象;5、尝试更新驱动程序,然后再次尝试创建或使用对象。

412

2023.10.16

俄罗斯Yandex引擎入口
俄罗斯Yandex引擎入口

2026年俄罗斯Yandex搜索引擎最新入口汇总,涵盖免登录、多语言支持、无广告视频播放及本地化服务等核心功能。阅读专题下面的文章了解更多详细内容。

158

2026.01.28

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 22.3万人学习

Django 教程
Django 教程

共28课时 | 3.6万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号