C++ RVO/NRVO以及move语义的影响

简介:

C++返回值优化和具名返回值优化是编译器的优化,在大多数情况下能提高性能,但是却难以受程序员控制。C++11中加入了move语义的支持,由此对RVO和NRVO会造成一定影响。下面以一段代码来说明。

RVO和NRVO在分别在copy/move construct,copy/move assignment八种简单情况,测试条件是g++ 4.8.2和clang++ 3.4,默认优化。

#include <iostream>
#include <vector>
#include <string>

struct Test {
    Test()
    {
        std::cout << "construct a Test object" << std::endl;
    }

    Test(const Test&)
    {
        std::cout << "copy construct  a Test object" << std::endl;
    }

    Test& operator=(const Test&)
    {
        std::cout << "copy assignment a Test object" << std::endl;
        return *this;
    }


    Test(Test&&)
    {
        std::cout << "move construct a Test object" << std::endl;
    }


    /*
    Test& operator=(Test &&t)
    {
        std::cout << "move assignment a Test object" << std::endl;
        return *this;
    }
    */

    ~Test()
    {
        std::cout << "destruct a Test object" << std::endl;
    }
};

Test getTest()
{
    return Test();
}

Test getTestWithName()
{
    Test temp;
    return temp;
}

int main()
{
    std::cout << "=============RVO==============" << std::endl; 
    std::cout << "++Test obj rvo for copy construct" << std::endl;
    auto obj1 = getTest();

    std::cout << "--------------" << std::endl;
    std::cout << "++Test obj rvo for move construct" << std::endl;
    auto obj111 = std::move(getTest());

    std::cout << "--------------" << std::endl;  
    std::cout << "++Test obj rvo for copy assignment" << std::endl;
    Test obj11; obj11 = getTest();

    std::cout << "--------------" << std::endl;
    std::cout << "++Test object rvo for move assignment" << std::endl;
    Test obj1111; obj1111 = std::move(getTest());


    std::cout << "=============NRVO==============" << std::endl; 
    std::cout << "++Test obj nrvo for copy construct" << std::endl;
    auto obj2 = getTestWithName();

    std::cout << "--------------" << std::endl;
    std::cout << "++Test obj nrvo for move construct" << std::endl;
    auto obj222 = std::move(getTestWithName());

    std::cout << "--------------" << std::endl;
    std::cout << "++Test obj nrvo for copy assignment" << std::endl;
    Test obj22; obj22 = getTestWithName();

    std::cout << "--------------" << std::endl;
    std::cout << "++Test obj nrvo for move assignment" << std::endl;
    Test obj2222; obj2222 = std::move(getTestWithName());

    std::cout << "==============================" << std::endl;
    // std::string s1 = "s1 string move semantics test", s2;
    //std::cout << "++before move s1\t" << s1 << std::endl;
    //s2 = std::move(s1);
    //std::cout << "++after move s1\t" << s1 << std::endl;
    //std::cout << "=============" << std::endl;
    return 0;
}

测试结果:

=============RVO==============
++Test obj rvo for copy construct
construct a Test object
--------------
++Test obj rvo for move construct
construct a Test object
move construct a Test object
destruct a Test object
--------------
++Test obj rvo for copy assignment
construct a Test object
construct a Test object
move assignment a Test object
destruct a Test object
--------------
++Test object rvo for move assignment
construct a Test object
construct a Test object
move assignment a Test object
destruct a Test object
=============NRVO==============
++Test obj nrvo for copy construct
construct a Test object
--------------
++Test obj nrvo for move construct
construct a Test object
move construct a Test object
destruct a Test object
--------------
++Test obj nrvo for copy assignment
construct a Test object
construct a Test object
move assignment a Test object
destruct a Test object
--------------
++Test obj nrvo for move assignment
construct a Test object
construct a Test object
move assignment a Test object
destruct a Test object
==============================
destruct a Test object
destruct a Test object
destruct a Test object
destruct a Test object
destruct a Test object
destruct a Test object
destruct a Test object
destruct a Test object

由此可得出几个简单结论:
1.copy construct本身在RVO和NRVO两种情况下被优化了,如果再加上move反而画蛇添足。
2.加入了move assignment后,默认是调用move assignment而不是copy assignment,可以将move assignment注释后测试。
3.对于RVO和NRVO来说,construction的情况编译器优化得比较好了,加入move语义主要是对于assignment有比较大影响

相关文章
|
6月前
|
算法 编译器 程序员
【C/C++ 解惑 】 std::move 将左值转换为右值的背后发生了什么?
【C/C++ 解惑 】 std::move 将左值转换为右值的背后发生了什么?
69 0
|
1月前
|
安全 编译器 程序员
【C++篇】C++类与对象深度解析(六):全面剖析拷贝省略、RVO、NRVO优化策略
【C++篇】C++类与对象深度解析(六):全面剖析拷贝省略、RVO、NRVO优化策略
46 2
|
5月前
|
编译器 C++ 开发者
C++一分钟之-返回值优化与Move Semantics
【6月更文挑战第19天】C++的RVO与移动语义提升效率,减少对象复制。RVO是编译器优化,避免临时对象的创建。移动语义通过右值引用和`std::move`转移资源所有权。注意RVO不是总有效,不应过度依赖。使用移动语义时,避免误用`std::move`导致对象无效。示例展示了RVO和移动构造函数的应用。理解并恰当使用这些机制能写出更高效代码。
66 3
|
6月前
|
编译器 C++ 容器
【C++11特性篇】探究【右值引用(移动语义)】是如何大大提高效率?——对比【拷贝构造&左值引用】
【C++11特性篇】探究【右值引用(移动语义)】是如何大大提高效率?——对比【拷贝构造&左值引用】
|
6月前
|
存储 安全 编译器
C++ std::move以及右值引用全面解析:从基础到实战,掌握现代C++高效编程
C++ std::move以及右值引用全面解析:从基础到实战,掌握现代C++高效编程
551 0
|
6月前
|
存储 编译器 C++
【C++】—— C++11新特性之 “右值引用和移动语义”
【C++】—— C++11新特性之 “右值引用和移动语义”
|
6月前
|
存储 编译器
C++11(左值(引用),右值(引用),移动语义,完美转发)
C++11(左值(引用),右值(引用),移动语义,完美转发)
60 0
|
6月前
|
编译器 C++
c++左值、右值引用和移动语义
c++左值、右值引用和移动语义
56 0
|
6月前
|
安全 编译器 C++
c++11新特性——右值引用和move语义
c++11新特性——右值引用和move语义