C++23中的std::expected如何处理错误?(类型安全的异常返回值处理)

发布时间 - 2026-01-23 00:00:00    点击率:
std::expected是C++23引入的值语义错误传播机制,不抛异常,以持有T或E的可变容器形式将错误视为可检查的合法状态,适用于需避免栈展开、精细控制错误流向的场景。

std::expected 是什么,它不替代 throw/catch

std::expected 是 C++23 引入的值语义错误传播机制,本质是一个持有 T(成功值)或 E(错误值)的可变容器。它**不抛异常**,也不捕获异常,而是把“出错”当作一种合法、可检查的数据状态来处理。适合那些你明确不想触发栈展开、需要细粒度控制错误流向的场景,比如系统调用封装、协议解析、配置加载。

如何从函数返回并检查 std::expected

典型模式是函数直接返回 std::expectedstd::expected<:string std::string> 这类类型。调用方必须显式检查是否含值:

std::expected parse_int(const char* s) {
    char* end;
    long v = std::strtol(s, &end, 10);
    if (*end != '\0' || errno == ERANGE) {
        return std::unexpected(std::errc::invalid_argument);
    }
    return static_cast(v);
}

auto result = parse_int("42"); if (result) { std::cout << "got: " << *result << "\n"; // OK } else { std::cout << "error: " << std::m

ake_error_code(result.error()).message() << "\n"; }
  • if (result) 检查是否为预期值(即 has_value()
  • *result 解包成功值(若未检查直接解包会 std::terminate
  • result.error() 获取错误对象,注意它返回的是 E&,不是拷贝
  • 不要对 std::expected*result —— 它没有值,只能用 result.has_value()!result

链式调用:and_then 和 or_else 的实际用途

当多个可能失败的操作需顺序执行时,and_then 可避免嵌套 if;而 or_else 用于提供备选路径(类似“默认值”或“fallback 函数”):

auto read_config() -> std::expected { /* ... */ }
auto validate(Config c) -> std::expected { /* ... */ }

auto pipeline = read_config() .and_then(validate) .and_then([](ValidConfig c) { return launch_service(c) ? std::expected{} : std::unexpected("launch failed"); });

if (!pipeline) { std::cerr << "Pipeline error: " << pipeline.error() << "\n"; }

  • and_then 接收一个返回 std::expected 的函数,自动扁平化嵌套结果(类似 std::optional::transform + std::expected 合并逻辑)
  • or_else 在当前为 error 时被调用,参数是 E&,返回另一个 std::expected —— 注意它不接收 std::unexpected,别写成 or_else([](auto u) { ... })
  • 所有这些操作都保持值语义,无异常、无动态内存分配(除非你的 TE 自身有)

和 std::optional、status code、异常混用时的坑

std::expected 不是万能胶。它和其它错误处理机制共存时容易混淆语义:

  • 别用 std::expected 模拟异常 —— 这违背了它的设计初衷,且丢失了栈信息;真要转发异常,请用 std::exception_ptr + 显式 std::rethrow_exception
  • 不要把 std::expected 当作 std::optional 用:后者表示“不存在”,前者表示“存在但失败”,语义不同
  • 与 C API 交互时,常见错误是把 errno 直接塞进 E 类型却不映射为标准错误码类别(如 std::errc),导致 std::error_code(e).message() 返回空字符串
  • 移动语义要注意:std::expected 移动后源对象进入未指定状态(C++23 标准未强制要求置空),不要在移动后访问其 value()error()

最常被忽略的一点:std::expectederror() 成员函数在 has_value() == true 时行为未定义 —— 不是返回垃圾值,而是直接 UB。检查必须前置。


# go  #   # ai  # c++  # String  # if  # 封装  # 成员函数  # throw  # Error  # auto  # 字符串  # errno  # bool  # int  # void  # 对象  # transform  # 链式  # 它不  # 的是  # 是一个  # 也不  # 多个  # 适用于  # 要把  # 要注意  # 这类 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 惠州网站建设制作推广,惠州市华视达文化传媒有限公司怎么样?  北京专业网站制作设计师招聘,北京白云观官方网站?  Laravel如何实现用户角色和权限系统_Laravel角色权限管理机制  jQuery validate插件功能与用法详解  香港服务器租用费用高吗?如何避免常见误区?  如何在万网自助建站中设置域名及备案?  UC浏览器如何切换小说阅读源_UC浏览器阅读源切换【方法】  JS经典正则表达式笔试题汇总  ,怎么在广州志愿者网站注册?  Laravel怎么定时执行任务_Laravel任务调度器Schedule配置与Cron设置【教程】  如何用PHP快速搭建高效网站?分步指南  在Oracle关闭情况下如何修改spfile的参数  Bootstrap CSS布局之列表  Python高阶函数应用_函数作为参数说明【指导】  图片制作网站免费软件,有没有免费的网站或软件可以将图片批量转为A4大小的pdf?  Laravel如何安装使用Debugbar工具栏_Laravel性能调试与SQL监控插件【步骤】  高端云建站费用究竟需要多少预算?  东莞专业网站制作公司有哪些,东莞招聘网站哪个好?  Windows Hello人脸识别突然无法使用  Win11怎么关闭资讯和兴趣_Windows11任务栏设置隐藏小组件  百度输入法全感官ai怎么关 百度输入法全感官皮肤关闭  php json中文编码为null的解决办法  如何安全更换建站之星模板并保留数据?  Laravel队列任务超时怎么办_Laravel Queue Timeout设置详解  米侠浏览器网页图片不显示怎么办 米侠图片加载修复  laravel怎么通过契约(Contracts)编程_laravel契约(Contracts)编程方法  ,网页ppt怎么弄成自己的ppt?  香港服务器网站生成指南:免费资源整合与高速稳定配置方案  Bootstrap整体框架之JavaScript插件架构  Laravel如何优化应用性能?(缓存和优化命令)  北京企业网站设计制作公司,北京铁路集团官方网站?  网站制作报价单模板图片,小松挖机官方网站报价?  php结合redis实现高并发下的抢购、秒杀功能的实例  Android利用动画实现背景逐渐变暗  高性能网站服务器部署指南:稳定运行与安全配置优化方案  如何做网站制作流程,*游戏网站怎么搭建?  奇安信“盘古石”团队突破 iOS 26.1 提权  Windows10电脑怎么查看硬盘通电时间_Win10使用工具检测磁盘健康  JS去除重复并统计数量的实现方法  JavaScript 输出显示内容(document.write、alert、innerHTML、console.log)  Laravel如何配置和使用缓存?(Redis代码示例)  PHP 实现电台节目表的智能时间匹配与今日/明日轮播逻辑  悟空浏览器如何设置小说背景色_悟空浏览器背景色设置【方法】  网站设计制作书签怎么做,怎样将网页添加到书签/主页书签/桌面?  JavaScript如何操作视频_媒体API怎么控制播放  Linux虚拟化技术教程_KVMQEMU虚拟机安装与调优  如何正确选择百度移动适配建站域名?  如何在万网主机上快速搭建网站?  如何在IIS中新建站点并解决端口绑定冲突?  Edge浏览器提示“由你的组织管理”怎么解决_去除浏览器托管提示【修复】