C++11中std::move、std::forward、左右值引用、移动构造函数的测试问题

 更新时间:2020年09月08日 08:27:46   作者:huik  
这篇文章主要介绍了C++11中std::move、std::forward、左右值引用、移动构造函数的测试,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

关于C++11新特性之std::move、std::forward、左右值引用网上资料已经很多了,我主要针对测试性能做一个测试,梳理一下这些逻辑,首先,左值比较熟悉,右值就是临时变量,意味着使用一次就不会再被使用了。针对这两种值引入了左值引用和右值引用,以及引用折叠的概念。

1.右值引用的举例测试

#include <iostream>
using namespace std;
​
//创建一个测试类
class A
{
public:
  A() : m_a(55)
  {
  }
​
  int m_a;
};
​
void funcA(A&& param) // 右值引用参数,只接受右值
{
  cout << param.m_a << endl; // param与a的地址一致,仅仅只是取了一个新名字
}
​
int main()
{
  A a;
  funcA(move(a)); //必须将其转换为右值
  cout << a.m_a << endl; //正常打印,所以std::move并没有移动的能力
  return 0;
}

2.左值和右值引用的举例测试,以及引出万能引用

构造一组重载函数,分别接受右值,和左值的参数,还有const A&的参数重载函数。

void funcA(const A& param)//既可以接受右值引用,也可以接受左值引用,但是有一个隐式转换const A&
void funcA(A& param)// 接受左值引用 
void funcA(A&& param) // 接受右值引用

const A& param既可以接受右值引用,也可以接受左值引用,但是存在一个隐式转换,const使用受限制。

#include <iostream>
using namespace std;
​
//创建一个测试类
class A
{
public:
  A() : m_a(55) // 构造函数
  {
    cout << "Constructor" << endl;
  }
  A(const A & other) : m_a(55) // copy构造函数
  {
    cout << "Copy Constructor" << endl;
    if (this == &other)
    {
      return;
    }
    this->m_a = other.m_a;
  }
  A& operator=(const A& other) // 赋值构造函数
  {
    cout << "= Constructor" << endl;
    if (this == &other)
    {
      return *this;
    }
    this->m_a = other.m_a;
    return *this;
  }
  int m_a;
};
void test(A&& pa) //测试是否为右值
{
  cout << "只接受右值" << endl;
}
void funcA(const A& param) // 既可以接受右值引用,也可以接受左值引用,但是有一个隐式转换const A&
{
  //test(param); //编译不过,param可以接受右值,但是param被转换为const左值
  //test(std::forward<A>(param)); //编译不过,param可以接受右值,但是param被转换为const左值
  cout << param.m_a << endl; 
}
void funcA(A& param) // 接受左值引用 
{
  //test(param); //编译不过,param可以接受右值,但是param被转换为左值
  test(std::forward<A>(param)); //编译通过,通过forward转发
  cout << param.m_a << endl;
}
void funcA(A&& param) // 接受右值引用
{
  //test(param); //编译不过,param被转换为左值
  test(std::forward<A>(param)); //编译通过,通过forward转发
  cout << param.m_a << endl;
}
​
int main()
{
  A a;
  const A& b = a;
  funcA(a);
  funcA(move(a));
  funcA(b);
  cout << a.m_a << endl; //正常打印,所以std::move并没有移动的能力
  return 0;
}

对此C++11引入了万能引用的概念,使得不需要那么多的重载函数,既可以接受右值引用,也可以接受左值引用。但是函数内部,再需要调用一个左值或者右值的函数时,我们就得需要forward模版类。

#include <iostream>
using namespace std;
​
//创建一个测试类
class A
{
public:
  A() : m_a(new int(55)) // 构造函数
  {
    cout << "Constructor" << endl;
  }
  A(const A & other) : m_a(new int(55)) // copy构造函数
  {
    cout << "Copy Constructor" << endl;
    if (this == &other)
      return;
    this->m_a = other.m_a;
  }
  A& operator=(const A& other) // 赋值构造函数
  {
    cout << "= Constructor" << endl;
    if (this == &other)
      return *this;
​
    this->m_a = other.m_a;
    return *this;
  }
  int* m_a;
};
void test(A&& pa) //测试是否为右值
{
  cout << "只接受右值" << endl;
}
void test(A& pa) //测试是否为左值
{
  cout << "只接受左值" << endl;
}
​
template<class T>
void funcA(T&& param)
{
  test(std::forward<T>(param)); //编译通过,通过forward完美转发
  cout << *param.m_a << endl;
}
​
int main()
{
  A a;
  funcA(a);
  funcA(move(a));
  cout << *a.m_a << endl; //正常打印,所以std::move并没有移动的能力
  return 0;
}

3.移动构造函数的引出

以上的所有特性,所能体现出来的是我们对于临时变量的使用,尽可能的使用中间生成的临时变量,提高性能,所谓的榨取最后的性能。移动构造函数注意的两点

1.调用移动构造函数时参数(被移动者)必须是右值。

2.调用移动构造函数后被移动者就不能再被使用。

#include <iostream>
using namespace std;
​
//创建一个测试类
class A
{
public:
  A() : m_a(new int(55)) // 构造函数
  {
    cout << "Constructor" << endl;
  }
  A(const A & other) : m_a(new int(55)) // copy构造函数
  {
    cout << "Copy Constructor" << endl;
    if (this == &other)
    {
      return;
    }
    this->m_a = other.m_a;
  }
  A& operator=(const A& other) // 赋值构造函数
  {
    cout << "= Constructor" << endl;
    if (this == &other)
    {
      return *this;
    }
    this->m_a = other.m_a;
    return *this;
  }
​
  A(A&& other) : m_a(other.m_a) // 移动构造函数,参数是一个右值,
  {
    cout << "Move Constructor" << endl;
    if (this == &other)
    {
      return;
    }
    other.m_a = nullptr; //移动后将被移动的对象数据清空
  }
​
  int* m_a;
};
void test(A&& pa) //测试是否为右值
{
  cout << "只接受右值" << endl;
}
void test(A& pa) //测试是否为左值
{
  cout << "只接受左值" << endl;
}
​
template<class T>
void funcA(T&& param)
{
  test(std::forward<T>(param)); //编译通过,通过forward完美转发
  cout << *param.m_a << endl;
}
​
int main()
{
  A a;
  funcA(a);
  funcA(move(a));
  A b(move(a)); //调用移动构造函数,新的对象是b对象
  cout << *a.m_a << endl; //数据已被移动,程序崩溃
  return 0;
}

移动构造函数一定程度上较少了临时内存的申请,减少不必要的拷贝,节省了空间和时间。以上特性在使用中还有很多需要注意的地方,如果我遇到了会及时的添加到这里,分享给大家,一起加油。

到此这篇关于C++11中std::move、std::forward、左右值引用、移动构造函数的测试问题的文章就介绍到这了,更多相关C++11中std::move、std 内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 手把手教你实现漂亮的Qt 登录界面

    手把手教你实现漂亮的Qt 登录界面

    最近在使用Qt5,Qt Creator做一个管理系统类的项目,需要用到登录界面,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-11-11
  • Linux下C语言的几道经典面试题小结(分享)

    Linux下C语言的几道经典面试题小结(分享)

    下面小编就为大家带来一篇Linux下C语言的几道经典面试题小结(分享)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-05-05
  • C++用指针变量作为函数的参数接受数组的值的问题详细总结

    C++用指针变量作为函数的参数接受数组的值的问题详细总结

    以下是对C++中用指针变量作为函数的参数接受数组的值的问题进行了详细的总结介绍,需要的朋友可以过来参考下,希望对大家有所帮助
    2013-10-10
  • 一篇文章带你了解C语言浮点数之间的比较规则

    一篇文章带你了解C语言浮点数之间的比较规则

    这篇文章主要介绍了魔性的float浮点数精度问题,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-08-08
  • C++ Qt之halcon读取像素项目过程详解

    C++ Qt之halcon读取像素项目过程详解

    这篇文章主要介绍了C++ Qt之halcon读取像素项目过程详解,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-09-09
  • Qt 使用Poppler实现pdf阅读器的示例代码

    Qt 使用Poppler实现pdf阅读器的示例代码

    下面小编就为大家分享一篇Qt 使用Poppler实现pdf阅读器的示例代码,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-01-01
  • 浅析C语言中的sizeof

    浅析C语言中的sizeof

    sizeof是C/C++中的一个操作符(operator),作用就是返回一个对象或者类型所占的内存字节数。返回值类型为size_t,在头文件stddef.h中定义
    2013-07-07
  • C语言将音视频时钟同步封装成通用模块的方法

    C语言将音视频时钟同步封装成通用模块的方法

    视频的时钟基于视频帧的时间戳,由于视频是通过一定的帧率渲染的,采用直接读取当前时间戳的方式获取时钟会造成一定的误差,精度不足,这篇文章主要介绍了c语言将音视频时钟同步封装成通用模块,需要的朋友可以参考下
    2022-09-09
  • C语言模拟实现动态通讯录

    C语言模拟实现动态通讯录

    本文主要介绍了C语言模拟实现动态通讯录,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-07-07
  • 深入理解Java事务的原理与应用

    深入理解Java事务的原理与应用

    下面小编就为大家带来一篇深入理解Java事务的原理与应用。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-06-06

最新评论