0

0

c++中的复制消除(Copy Elision)是什么_c++ RVO/NRVO优化详解【编译器】

裘德小鎮的故事

裘德小鎮的故事

发布时间:2025-12-22 17:36:51

|

783人浏览过

|

来源于php中文网

原创

复制消除是C++编译器跳过不必要的对象复制或移动的优化技术;C++17起RVO强制实施,NRVO仍为鼓励而非强制,常见于返回局部对象、临时对象直接初始化等场景。

c++中的复制消除(copy elision)是什么_c++ rvo/nrvo优化详解【编译器】

复制消除(Copy Elision)是 C++ 编译器在满足语义等价的前提下,**跳过不必要的对象复制或移动操作**的一种优化技术。它不是“可选的优化”,而是从 C++17 起被强制要求实现的规则(特别是 RVO 和 NRVO 场景),能直接避免临时对象构造、析构及拷贝/移动函数调用,提升性能且可能改变程序行为(比如绕过有副作用的拷贝构造函数)。

什么是 RVO(Return Value Optimization)?

RVO 指编译器对**按值返回局部对象**时做的优化:不先构造局部对象再拷贝回返回值,而是直接在调用方为返回值准备的内存位置上构造该对象。

例如:

有道翻译AI助手
有道翻译AI助手

有道翻译提供即时免费的中文、英语、日语、韩语、法语、德语、俄语、西班牙语、葡萄牙语、越南语、印尼语、意大利语、荷兰语、泰语全文翻译、网页翻译、文档翻译、PDF翻

下载
A createA() {
    return A(42); // 编译器可直接在 caller 的返回槽中构造 A(42)
}
A a = createA(); // 不调用 A 的拷贝/移动构造函数

关键点:

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

  • 仅适用于返回一个非 volatile 的具名或匿名临时对象
  • 函数返回类型必须与局部对象类型相同(不能发生隐式转换);
  • C++17 起,这种情形下的复制消除是强制的(即使拷贝/移动构造函数有副作用,也必须省略)。

什么是 NRVO(Named Return Value Optimization)?

NRVO 是 RVO 的扩展,允许编译器对**返回一个具名局部变量**的情况也进行消除——只要该变量在所有 return 语句中都被返回,且类型匹配。

例如:

A createA(bool flag) {
    A a1(1), a2(2);
    if (flag) return a1; // 可能触发 NRVO
    else      return a2; // 同样可能触发 NRVO
}

注意:

  • NRVO 不是强制的(C++17 仍属“鼓励但不保证”),不同编译器、不同优化等级下表现可能不同;
  • 多个 return 路径返回不同变量、或存在中间赋值/修改,会显著降低 NRVO 成功率;
  • 启用 -O2-O3 通常能提高 NRVO 触发概率。

复制消除发生的典型场景

除 RVO/NRVO 外,C++17 还强制规定了另一类复制消除:临时对象绑定到引用时的初始化(即所谓 “Temporary Materialization Elimination” 的一部分),以及更广义的“复制/移动省略”场景:

  • A a = A(123); → 直接构造 a,跳过拷贝/移动(C++17 强制);
  • func(A(456));(func 参数为值传递)→ 临时对象直接在 func 参数内存中构造;
  • throw/catch 中抛出并捕获同类型对象时,也可能省略复制(依赖实现,非强制)。

⚠️ 注意:这些优化都要求拷贝/移动构造函数**可访问且不被显式删除**(否则编译失败),但实际是否调用它们,由编译器决定(C++17 前)或必须不调用(C++17 RVO/NRVO 场景)。

如何确认复制消除是否发生?

最可靠方式是观察构造/析构/拷贝函数的调用次数(加打印或断点),但要注意:

  • 开启优化(如 -O2)后才大概率触发;
  • Debug 模式(-O0)通常禁用 RVO/NRVO,便于调试;
  • 使用 [[nodiscard]]std::move 并不会“帮助”触发 RVO,反而可能阻碍(比如 return std::move(local); 会禁止 NRVO);
  • Clang/GCC 提供 -fno-elide-constructors 强制关闭该优化,用于测试副作用逻辑。

基本上就这些。理解复制消除,不只是为了性能,更是为了写出符合现代 C++ 语义预期的代码——尤其当你的拷贝构造函数有日志、计数、资源申请等副作用时,它可能根本不会被调用。

相关专题

更多
c++中volatile关键字的作用
c++中volatile关键字的作用

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

67

2025.10.23

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

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

106

2024.02.23

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

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

106

2024.02.23

C++ 单元测试与代码质量保障
C++ 单元测试与代码质量保障

本专题系统讲解 C++ 在单元测试与代码质量保障方面的实战方法,包括测试驱动开发理念、Google Test/Google Mock 的使用、测试用例设计、边界条件验证、持续集成中的自动化测试流程,以及常见代码质量问题的发现与修复。通过工程化示例,帮助开发者建立 可测试、可维护、高质量的 C++ 项目体系。

5

2026.01.16

java数据库连接教程大全
java数据库连接教程大全

本专题整合了java数据库连接相关教程,阅读专题下面的文章了解更多详细内容。

26

2026.01.15

Java音频处理教程汇总
Java音频处理教程汇总

本专题整合了java音频处理教程大全,阅读专题下面的文章了解更多详细内容。

12

2026.01.15

windows查看wifi密码教程大全
windows查看wifi密码教程大全

本专题整合了windows查看wifi密码教程大全,阅读专题下面的文章了解更多详细内容。

35

2026.01.15

浏览器缓存清理方法汇总
浏览器缓存清理方法汇总

本专题整合了浏览器缓存清理教程汇总,阅读专题下面的文章了解更多详细内容。

5

2026.01.15

ps图片相关教程汇总
ps图片相关教程汇总

本专题整合了ps图片设置相关教程合集,阅读专题下面的文章了解更多详细内容。

8

2026.01.15

热门下载

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

精品课程

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

共94课时 | 6.8万人学习

C 教程
C 教程

共75课时 | 4万人学习

C++教程
C++教程

共115课时 | 12.3万人学习

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

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