0

0

C++中如何利用std::exchange实现异常安全的赋值?(代码健壮性)

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-02-28 13:05:03

|

519人浏览过

|

来源于php中文网

原创

std::exchange提升异常安全性因其将取旧值与设新值合并为原子操作,避免手动两步序列中赋值失败导致原始状态不可恢复;它等价于先move再赋值,中间无用户可控状态,适用于移动/拷贝类型,仅当新值构造或赋值可能抛异常时价值凸显。

c++中如何利用std::exchange实现异常安全的赋值?(代码健壮性)

std::exchange 为什么能提升异常安全性?

因为 std::exchange 把“取旧值”和“设新值”合并成一个原子操作,避免了手动写 old = x; x = new_val; 这种两步序列——万一 x = new_val 抛异常,old 已经被取走,原始状态不可恢复。

它底层等价于:auto old = std::move(x); x = std::forward<t>(new_val); return old;</t>,关键在 std::move(x) 和赋值之间没有用户可控的中间状态。

  • 适用于所有支持移动或拷贝的类型,但只有当新值构造/赋值可能抛异常时,它的安全价值才真正体现
  • 如果 T 的移动构造/赋值是 noexcept(比如 intstd::unique_ptr),那 std::exchange 本身也基本不会抛异常
  • 注意:它不保证整个函数异常安全,只保证这一步“读旧+写新”的完整性

用在 swap 或资源替换场景时怎么写才不出错?

典型误用是把它当普通赋值替代品,忽略返回值或类型匹配问题。比如交换两个成员,不能只写 std::exchange(a, b) 就完事——它返回的是 a 的旧值,不是 b 的旧值。

正确做法是配合临时变量或结构化绑定:

FreeTTS
FreeTTS

FreeTTS是一个免费开源的在线文本到语音生成解决方案,可以将文本转换成MP3,

下载

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

auto old_a = std::exchange(a, b);  // a 变成 b,old_a 是原来的 a
b = std::move(old_a);               // b 变成原来的 a
  • ab 类型不同(如 std::stringconst char*),std::exchange(a, "hello") 会隐式转换,但可能触发非预期的构造(比如抛 std::bad_alloc),这时要确认右侧是否 noexcept
  • 在 RAII 对象中替换内部指针时,必须确保新指针构造不抛异常,否则旧资源可能已丢失(std::exchange(ptr, new T)new T 若抛异常,ptr 已置为 nullptr

和直接赋值比,性能和 ABI 兼容性有啥实际影响?

绝大多数情况下没区别:std::exchange(x, v) 编译后和 { auto t = std::move(x); x = std::move(v); return t; } 几乎一致,现代编译器能很好优化。

  • 唯一可测差异出现在 v 是右值且类型有开销较大的移动构造时——std::exchange 强制移动两次(一次取旧,一次设新),而手写可能省一次(比如先 swap
  • ABI 上完全无影响:它只是个内联函数模板,不改变对象布局或调用约定
  • 不要在热循环里为“显得更安全”盲目替换所有赋值;只在明确需要捕获旧值 + 保证赋值原子性的地方用

容易被忽略的 const 与引用陷阱

std::exchange 要求第一个参数是非 const 的左值引用,传 const 对象或临时量会编译失败,错误信息往往指向模板推导而非语义问题。

常见报错:error: no matching function for call to 'exchange(const std::string&, std::string)'

  • 不能对 const 成员变量使用,除非该成员本身是 mutable(极少见)
  • 不能用于 std::vector::at(i) 这类返回临时量的表达式(std::exchange(v.at(0), x) 错误:不是可修改的左值)
  • 如果目标是 std::optional<t></t> 的值,得写 std::exchange(opt.value(), new_val),而不是 std::exchange(opt, std::nullopt)——后者交换的是整个 optional,语义完全不同
事情说清了就结束

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
string转int
string转int

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

870

2023.08.02

scripterror怎么解决
scripterror怎么解决

scripterror的解决办法有检查语法、文件路径、检查网络连接、浏览器兼容性、使用try-catch语句、使用开发者工具进行调试、更新浏览器和JavaScript库或寻求专业帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

391

2023.10.18

500error怎么解决
500error怎么解决

500error的解决办法有检查服务器日志、检查代码、检查服务器配置、更新软件版本、重新启动服务、调试代码和寻求帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

349

2023.10.25

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

556

2023.09.20

string转int
string转int

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

870

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

592

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

294

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

210

2025.08.29

Golang 测试体系与代码质量保障:工程级可靠性建设
Golang 测试体系与代码质量保障:工程级可靠性建设

Go语言测试体系与代码质量保障聚焦于构建工程级可靠性系统。本专题深入解析Go的测试工具链(如go test)、单元测试、集成测试及端到端测试实践,结合代码覆盖率分析、静态代码扫描(如go vet)和动态分析工具,建立全链路质量监控机制。通过自动化测试框架、持续集成(CI)流水线配置及代码审查规范,实现测试用例管理、缺陷追踪与质量门禁控制,确保代码健壮性与可维护性,为高可靠性工程系统提供质量保障。

0

2026.02.28

热门下载

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

精品课程

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

共94课时 | 10.3万人学习

C 教程
C 教程

共75课时 | 5万人学习

C++教程
C++教程

共115课时 | 19.7万人学习

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

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