移动语义然后复制c ++

时间:2017-10-16 10:12:29

标签: c++11 vector move

我写了一个比较移动和测试语义的小测试:

#include <vector>
#include <iostream>
#include <iterator>
#include <chrono>
#include <iomanip>

using namespace std;

int main()
{

    int lenLeft = 3;
    int lenMid = 3;
    vector<int> lenVec{10,100,1000,static_cast<int>(1e4),static_cast<int>(1e5),static_cast<int>(1e6),static_cast<int>(1e7)};
    int reps = 100;
    vector<double>delta_t_move;
    vector<double>delta_t_copy;


    //move
    cout<<"move"<<endl;
    {

        for(int len : lenVec)
        {
            auto startTime = std::chrono::high_resolution_clock::now();

            for(int i = 0; i<reps;i++)
            {
                vector<int> leftVec(len,0);

                vector<int> rightVec;
                move(leftVec.begin()+lenLeft+lenMid,leftVec.end(),std::back_inserter(rightVec));
                leftVec.erase(leftVec.begin()+lenLeft+lenMid,leftVec.end());

                vector<int> midVec;
                std::move(leftVec.begin()+lenLeft,leftVec.begin()+lenLeft+lenMid,std::back_inserter(midVec));
                leftVec.erase(leftVec.begin()+lenLeft,leftVec.begin()+lenLeft+lenMid);
            }

            auto endTime = std::chrono::high_resolution_clock::now();
            std::chrono::duration<double> elapsed = endTime - startTime;
            delta_t_move.push_back(elapsed.count());
        }
    }

    //copy
    cout<<"copy"<<endl;
    {

        for(int len : lenVec)
        {

            auto startTime = std::chrono::high_resolution_clock::now();

            for(int i = 0; i<reps;i++)
            {
                vector<int> leftVec(len,0);

                vector<int> rightVec = vector<int>(leftVec.begin()+lenLeft+lenMid,leftVec.end());
                leftVec.erase(leftVec.begin()+lenLeft+lenMid,leftVec.end());

                vector<int> midVec = vector<int>(leftVec.begin()+lenLeft,leftVec.begin()+lenLeft+lenMid);
                leftVec.erase(leftVec.begin()+lenLeft,leftVec.begin()+lenLeft+lenMid);
            }

            auto endTime = std::chrono::high_resolution_clock::now();
            std::chrono::duration<double> elapsed = endTime - startTime;
            delta_t_copy.push_back(elapsed.count());
        }
    }

    for(int i = 0; i<lenVec.size();i++)
    {
        cout<<"lenVec = "<<setw(40)<<lenVec.at(i)<<"\t\t : delta_t_copy/delta_t_move = "<< delta_t_copy.at(i)/delta_t_move.at(i)<<endl;
    }


    return 0;
}

我得到的这个程序的输出是:

move
copy
lenVec = 10 : delta_t_copy/delta_t_move = 0.431172
lenVec = 100 : delta_t_copy/delta_t_move = 0.257102
lenVec = 1000 : delta_t_copy/delta_t_move = 0.166006
lenVec = 10000 : delta_t_copy/delta_t_move = 0.108573
lenVec = 100000 : delta_t_copy/delta_t_move = 0.113769
lenVec = 1000000 : delta_t_copy/delta_t_move = 0.134912
lenVec = 10000000 : delta_t_copy/delta_t_move = 0.133874

我将大小为len的初始矢量分成3个部分。第一段长度为3,中间长度为3,其余为长度len-6。 我的结果表明,复制语义比移动语义快得多。

我正在使用MSVC2015。

知道这是怎么回事?什么情况下移动语义更快?

2 个答案:

答案 0 :(得分:1)

您的基准是有缺陷且不精确的。

  • 首先,您从未提及您正在使用的编译器标志 - 您是否启用了优化?

  • 最大的问题是您使用的是int元素的向量。 int移动在效果方面相当于int副本。

  • 要获得正确而完整的基准测试,您还应该尝试使用g ++和clang ++。

  • 您没有在任何地方致电std::vector<T>::reserve

  • 虽然不太可能,但无法保证编译器不会积极优化您的循环(例如,展开或融合多个循环)

您还在“复制”基准测试中执行了一些不必要的操作:

vector<int> rightVec = vector<int>(leftVec.begin()+lenLeft+lenMid,leftVec.end());    
vector<int> midVec = vector<int>(leftVec.begin()+lenLeft,leftVec.begin()+lenLeft+lenMid);

为什么不呢......

vector<int> rightVec(leftVec.begin()+lenLeft+lenMid,leftVec.end());    
vector<int> midVec(leftVec.begin()+lenLeft,leftVec.begin()+lenLeft+lenMid);

...

答案 1 :(得分:1)

除了Vittorio's answer之外,我还要指出导致"move"代码路径性能下降的具体原因。

你的基准归结为这种填充矢量的方式之间的区别:

vector<int> rightVec = vector<int>(startIt, endIt);

与这种填充方式相比:

vector<int> rightVec;
move(startIt, endIt, std::back_inserter(rightVec));

因为这是 only 部分,其中两个代码路径差异很大。

预计第二个版本会因两个原因而变慢:

  • 在第二种情况下,目标向量不知道它应该存储多少元素,因此在执行插入时它必须保持增长。生长的向量是昂贵的,因为它涉及重新分配和复制/移动先前插入的元素。您可以在reseve()之前插入适当的move来阻止此缺点。这将大大降低此代码路径上的性能损失。
  • 剩下的小的性能差异将归因于您移动到back_inserter这导致以元素方式插入目标向量的事实,就像在第一种情况下执行的批量插入一样。

如果你注意减轻这两点的影响,你会发现运行时大致相同,因为正如已经指出的那样,移动和复制是int元素的等效操作。