0

0

c++26的Rethrowing exceptions with more context提案是什么? (改进异常诊断)

php中文网

php中文网

发布时间:2026-01-12 13:48:08

|

892人浏览过

|

来源于php中文网

原创

C++26 不包含 rethrow_with_context,它仍处于 TS 探索阶段;标准库不支持异常上下文是因值语义、零开销原则及跨 ABI 兼容性限制;当前可靠方案是手动包装异常并结合 std::source_location 与日志。

c++26的rethrowing exceptions with more context提案是什么? (改进异常诊断)

C++26 的 rethrow_with_context 提案(P2697R3)尚未被批准为标准特性,目前仍处于 TS 探索阶段,且不会出现在 C++26 中。它不是已定稿的 C++26 功能,而是 WG21 讨论中的一个实验性方向——目标是让异常在传播途中能自动附加上下文信息(如函数名、源码位置、局部变量快照等),从而改善调试体验。

为什么标准库至今不支持异常上下文?

C++ 异常对象本身是值语义的:抛出时复制/移动,捕获时绑定到 catch 参数。标准要求异常对象必须满足 CopyConstructible(或 C++11 后的 MoveConstructible),而任意“上下文”(比如帧快照、std::source_location 实例、甚至 std::string)会破坏零开销抽象原则,并引入隐式动态分配和 ABI 不稳定性。

更关键的是:异常可能跨 shared library 边界传播,而上下文数据的内存布局、生命周期、甚至 std::string 的实现细节(SSO vs 堆分配)在不同编译器/标准库间不兼容,导致 catch 侧访问损坏内存或崩溃。

std::exception_ptr 是唯一可移植的“异常转发”机制

如果你需要在异常传播链中插入诊断信息,目前最可靠的方式是手动包装:

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

Illustroke
Illustroke

text to SVG,AI矢量插画生成工具

下载
  • std::make_exception_ptr 捕获原始异常
  • 构造一个自定义异常类型(如 contextual_error),在其构造函数中保存 std::source_location::current()、关键变量值、以及原始 std::exception_ptr
  • 在合适位置 throw 这个新异常
  • 下游 catch 可递归调用 std::rethrow_exception 获取原始异常
struct contextual_error : std::runtime_error {
    std::exception_ptr cause;
    std::source_location loc;
contextual_error(const char* msg, std::exception_ptr p, 
                 std::source_location l = std::source_location::current())
    : std::runtime_error(msg), cause(std::move(p)), loc(l) {}

const char* what() const noexcept override {
    return std::runtime_error::what();
}

};

// 使用示例 try { risky_operation(); } catch (...) { auto ep = std::current_exception(); throw contextual_error("failed in process_user_data", std::move(ep)); }

当前实际可用的诊断替代方案

不要依赖未落地的提案,优先采用已被广泛验证的手段:

  • std::source_location(C++20):在每个关键 throw 点显式记录位置,无需运行时开销
  • 日志宏(如 LOG_ERROR("at {}:{}: {}", loc.file_name(), loc.line(), e.what())):在 catch 块中立即输出上下文,比“重抛带上下文”更可控、无 ABI 风险
  • 调试器集成:LLDB/GDB 支持 catch throwbt 查看完整栈,配合 DWARF 信息比运行时注入更准确
  • 静态分析工具(如 clang-tidy 的 cppcoreguidelines-avoid-goto 类规则):提前发现异常路径中的资源泄漏点

C++ 对异常上下文的谨慎态度,本质是坚持“你不需要为不用的功能付费”。任何试图绕过这一原则的运行时机制,都会在跨模块、跨编译器、跨优化等级时暴露脆弱性。真正可靠的诊断,永远建立在明确控制权移交(catch → 日志 → 包装 → throw)和编译期可验证的信息(source_location)之上。

相关文章

Windows激活工具
Windows激活工具

Windows激活工具是正版认证的激活工具,永久激活,一键解决windows许可证即将过期。可激活win7系统、win8.1系统、win10系统、win11系统。下载后先看完视频激活教程,再进行操作,100%激活成功。

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

315

2023.08.02

go语言goto的用法
go语言goto的用法

本专题整合了go语言goto的用法,阅读专题下面的文章了解更多详细内容。

133

2025.09.05

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

386

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

569

2023.08.10

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

386

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

569

2023.08.10

c++主流开发框架汇总
c++主流开发框架汇总

本专题整合了c++开发框架推荐,阅读专题下面的文章了解更多详细内容。

78

2026.01.09

c++框架学习教程汇总
c++框架学习教程汇总

本专题整合了c++框架学习教程汇总,阅读专题下面的文章了解更多详细内容。

45

2026.01.09

学python好用的网站推荐
学python好用的网站推荐

本专题整合了python学习教程汇总,阅读专题下面的文章了解更多详细内容。

118

2026.01.09

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 3.6万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.8万人学习

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

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