【C++】RVO、NRVO优化以及返回值优化失效的场景

2023-12-15 10:04

本文主要是介绍【C++】RVO、NRVO优化以及返回值优化失效的场景,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

    • **简单对象的返回**
    • **多个返回语句**
    • **具有可观察副作用的对象**

(RVO, Return Value Optimization,返回值优化,或者NRVO,Named Return Value optimization)。

使用-fno-elide-constructors选项可以在g++/clang++中关闭这个优化
但若在编译的时候不使用该选项的话,很多构造和移动都被省略了。对于下面这样的代码,一旦打开g++/clang++的RVO/NRVO,从ReturnValue函数中a变量拷贝/移动构造临时变量,以及从临时变量拷贝/移动构造b的二重奏就通通没有了

        A ReturnRvalue() { A a(); return a; }A b = ReturnRvalue();

b变量实际就使用了ReturnRvalue函数中a的地址,任何的拷贝和移动都没有了。通俗地说,就是b变量直接“霸占”了a变量。这是编译器中一个效果非常好的一个优化。不过RVO/NRVO并不是对任何情况都有效。比如有些情况下,一些构造是无法省略的。还有一些情况,即使RVO/NRVO完成了,也不能达到最好的效果。但结论是明显的,移动语义可以解决编译器无法解决的优化问题,因而总是有用的。

返回值优化(Return Value Optimization,RVO)是一种编译器优化技术,旨在减少函数返回对象的副本构造和析构成本。通过RVO,编译器可以直接将局部对象放置在函数调用方所期望的位置,而不是创建一个临时副本。

尽管RVO在现代编译器中通常会生效,但并不能保证它始终有效。以下是一些情况下RVO可能无法生效的情况:

  1. 多个返回语句:如果函数中存在多个返回语句,且每个返回语句返回的对象类型不同,编译器可能无法进行RVO。这是因为编译器需要确保能够正确地创建要返回的对象,并且无法预先确定哪个对象将最终被返回。
  2. 异常处理:当函数中存在异常处理代码时,编译器可能会禁用RVO。这是为了确保在异常抛出时能够捕获、处理或传递异常对象的拷贝。
  3. 对象具有可观察的副作用:如果对象的拷贝构造函数、析构函数或移动构造函数具有可观察的副作用,编译器可能会禁用RVO。这是为了确保对象的副本构造和析构过程按照预期进行,避免对程序的行为产生副作用。
  4. C++标准规定:虽然大多数现代编译器支持RVO,但C++标准并不要求编译器实现该优化。因此,特定编译器或编译器版本可能在某些情况下选择不执行RVO。

需要注意的是,无论RVO是否生效,代码的行为应始终符合C++语言中对拷贝构造函数、析构函数和移动构造函数的要求。如果有特定的性能需求或对RVO的可靠性有疑问,可以通过使用移动语义、返回指针或引用等其他技术来显式控制对象的生命周期和复制行为。

简单对象的返回

#include <iostream>struct MyObject {int value;MyObject(int val) : value(val) {std::cout << "Constructor called\n";}MyObject(const MyObject& other) : value(other.value) {std::cout << "Copy constructor called\n";}~MyObject() {std::cout << "Destructor called\n";}
};MyObject createObject() {MyObject obj = MyObject(42);return obj;
}int main() {MyObject obj = createObject();std::cout << "Value: " << obj.value << "\n";return 0;
}
Constructor called
Value: 42         
Destructor called 

多个返回语句

#include <iostream>struct MyObject {int value;MyObject(int val) : value(val) {std::cout << "Constructor called\n";}MyObject(const MyObject& other) : value(other.value) {std::cout << "Copy constructor called\n";}~MyObject() {std::cout << "Destructor called\n";}
};MyObject createObject(bool flag) {MyObject obj1 = MyObject(42);MyObject obj2 = MyObject(43);if (flag) {return obj1;} else {return obj2;}
}int main() {MyObject obj = createObject(true);std::cout << "Value: " << obj.value << "\n";return 0;
}
Constructor called     
Constructor called     
Copy constructor called
Destructor called      
Destructor called      
Value: 42              
Destructor called   

在这个例子中,createObject() 函数根据 flag 参数返回不同的对象。由于无法预测哪个对象将被返回,编译器无法进行RVO。因此,在每个返回语句处都会调用相应的拷贝构造函数

具有可观察副作用的对象

#include <iostream>struct MyObject {int value;MyObject(int val) : value(val) {std::cout << "Constructor called\n";}MyObject(const MyObject& other) : value(other.value) {std::cout << "Copy constructor called\n";}~MyObject() {std::cout << "Destructor called\n";}
};MyObject createObject(bool flag) {MyObject obj1(42);if (flag) {return obj1;} else {return MyObject(64);}
}int main() {MyObject obj = createObject(true);std::cout << "Value: " << obj.value << "\n";return 0;
}

在这个示例中,createObject() 函数返回一个对象,但它也引用了一个局部对象 obj1。由于该对象在返回之后保持有效性,编译器无法执行RVO,因此会调用拷贝构造函数以创建返回值。

所以如果有这种情况的话,最好为你的资源类提供一个移动构造函数。

下面是GPT总结的一些经验,用于帮助我们判断RVO是否会生效.

确切确定某段代码是否会触发返回值优化(RVO)可能是一项具有挑战性的任务,因为它依赖于编译器的实现和优化策略。然而,以下是一些总结性的提示,可以帮助您判断代码中是否可能不会发生RVO:

  1. 当函数有多个返回语句时,尤其是返回不同类型的对象。
  2. 当函数的返回值依赖于外部变量或传递给函数的参数。
  3. 当函数内部使用了异常处理机制。
  4. 当拷贝构造函数、析构函数或移动构造函数有可观察的副作用,例如打印输出、修改全局状态等。
  5. 当函数返回指向局部对象的指针或引用。

这些情况下,编译器可能无法进行返回值优化,因为它无法安全地将局部对象直接放置在函数调用方所期望的位置。

这篇关于【C++】RVO、NRVO优化以及返回值优化失效的场景的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/496063

相关文章

Windows下C++使用SQLitede的操作过程

《Windows下C++使用SQLitede的操作过程》本文介绍了Windows下C++使用SQLite的安装配置、CppSQLite库封装优势、核心功能(如数据库连接、事务管理)、跨平台支持及性能优... 目录Windows下C++使用SQLite1、安装2、代码示例CppSQLite:C++轻松操作SQ

C++中RAII资源获取即初始化

《C++中RAII资源获取即初始化》RAII通过构造/析构自动管理资源生命周期,确保安全释放,本文就来介绍一下C++中的RAII技术及其应用,具有一定的参考价值,感兴趣的可以了解一下... 目录一、核心原理与机制二、标准库中的RAII实现三、自定义RAII类设计原则四、常见应用场景1. 内存管理2. 文件操

C++中零拷贝的多种实现方式

《C++中零拷贝的多种实现方式》本文主要介绍了C++中零拷贝的实现示例,旨在在减少数据在内存中的不必要复制,从而提高程序性能、降低内存使用并减少CPU消耗,零拷贝技术通过多种方式实现,下面就来了解一下... 目录一、C++中零拷贝技术的核心概念二、std::string_view 简介三、std::stri

C++高效内存池实现减少动态分配开销的解决方案

《C++高效内存池实现减少动态分配开销的解决方案》C++动态内存分配存在系统调用开销、碎片化和锁竞争等性能问题,内存池通过预分配、分块管理和缓存复用解决这些问题,下面就来了解一下... 目录一、C++内存分配的性能挑战二、内存池技术的核心原理三、主流内存池实现:TCMalloc与Jemalloc1. TCM

C++ 函数 strftime 和时间格式示例详解

《C++函数strftime和时间格式示例详解》strftime是C/C++标准库中用于格式化日期和时间的函数,定义在ctime头文件中,它将tm结构体中的时间信息转换为指定格式的字符串,是处理... 目录C++ 函数 strftipythonme 详解一、函数原型二、功能描述三、格式字符串说明四、返回值五

MyBatis Plus 中 update_time 字段自动填充失效的原因分析及解决方案(最新整理)

《MyBatisPlus中update_time字段自动填充失效的原因分析及解决方案(最新整理)》在使用MyBatisPlus时,通常我们会在数据库表中设置create_time和update... 目录前言一、问题现象二、原因分析三、总结:常见原因与解决方法对照表四、推荐写法前言在使用 MyBATis

Python主动抛出异常的各种用法和场景分析

《Python主动抛出异常的各种用法和场景分析》在Python中,我们不仅可以捕获和处理异常,还可以主动抛出异常,也就是以类的方式自定义错误的类型和提示信息,这在编程中非常有用,下面我将详细解释主动抛... 目录一、为什么要主动抛出异常?二、基本语法:raise关键字基本示例三、raise的多种用法1. 抛

C++作用域和标识符查找规则详解

《C++作用域和标识符查找规则详解》在C++中,作用域(Scope)和标识符查找(IdentifierLookup)是理解代码行为的重要概念,本文将详细介绍这些规则,并通过实例来说明它们的工作原理,需... 目录作用域标识符查找规则1. 普通查找(Ordinary Lookup)2. 限定查找(Qualif

Spring组件实例化扩展点之InstantiationAwareBeanPostProcessor使用场景解析

《Spring组件实例化扩展点之InstantiationAwareBeanPostProcessor使用场景解析》InstantiationAwareBeanPostProcessor是Spring... 目录一、什么是InstantiationAwareBeanPostProcessor?二、核心方法解

Java 枚举的基本使用方法及实际使用场景

《Java枚举的基本使用方法及实际使用场景》枚举是Java中一种特殊的类,用于定义一组固定的常量,枚举类型提供了更好的类型安全性和可读性,适用于需要定义一组有限且固定的值的场景,本文给大家介绍Jav... 目录一、什么是枚举?二、枚举的基本使用方法定义枚举三、实际使用场景代替常量状态机四、更多用法1.实现接