首页 > 编程语言 >C++11:移动语义

C++11:移动语义

时间:2022-09-28 18:00:33浏览次数:74  
标签:11 tmp cout int 语义 C++ Test 转移 构造函数


为什么需要移动语义

#include <iostream>
using namespace std;

class Test
{
public:
Test(int a = 0)
{//普通构造函数
d = new int(a);
cout << "构造函数\n";
}

Test(const Test & tmp)
{//拷贝构造函数
d = new int;
*d = *(tmp.d);
cout << "拷贝构造函数\n";
}

~Test()
{//析构函数
if(d != NULL)
{
delete d;
cout << "delete d\n";
}
cout << "析构函数\n";
}

int * d;
};

Test GetTmp()
{
Test h;
cout << "Resource from " << __func__ << ": " << (void *)h.d << endl;
return h;
}

int main()
{
Test obj = GetTmp();
cout << "Resource from " << __func__ << ": " << (void *)obj.d << endl;

return 0;
}

编译运行结果:

C++11:移动语义_右值引用

编译器会对返回值进行优化,简称RVO,是编译器的一项优化技术,它涉及(功能是)消除为保存函数返回值而创建的临时对象。

-fno-elide-constructors,此选项作用是,在 g++ 上编译时关闭 RVO。

通过上面的例子看到,临时对象的维护 ( 创建和销毁 ) 对性能有严重影响

右值引用是用来支持转移语义的。转移语义可以将资源 ( 堆,系统对象等 ) 从一个对象转移到另一个对象,这样能够减少不必要的临时对象的创建、拷贝以及销毁,能够大幅度提高 C++ 应用程序的性能。

转移语义是和拷贝语义相对的,可以类比文件的剪切与拷贝,当我们将文件从一个目录拷贝到另一个目录时,速度比剪切慢很多。

通过转移语义,临时对象中的资源能够转移其它的对象里。

移动语义定义

在现有的 C++ 机制中,我们可以定义拷贝构造函数和赋值函数。要实现转移语义,需要定义转移构造函数,还可以定义转移赋值操作符。对于右值的拷贝和赋值会调用转移构造函数和转移赋值操作符。

如果转移构造函数和转移拷贝操作符没有定义,那么就遵循现有的机制,拷贝构造函数和赋值操作符会被调用。

普通的函数和操作符也可以利用右值引用操作符实现转移语义。

转移构造函数

#include <iostream>
using namespace std;

class Test
{
public:
Test(int a = 0)
{
d = new int(a);
cout << "构造函数\n";
}

Test(const Test & tmp)
{
d = new int;
*d = *(tmp.d);
cout << "拷贝构造函数\n";
}
#if 1
Test(Test && tmp)
{ // 移动构造函数
d = tmp.d;
tmp.d = NULL; // 将临时值的指针成员置空
cout << "移动构造函数" << endl;
}
#endif

~Test()
{
if(d != NULL)
{
delete d;
cout << "delete d\n";
}
cout << "析构函数\n";
}

int * d;
};

Test GetTmp()
{
Test h;
cout << "Resource from " << __func__ << ": " << (void *)h.d << endl;
return h;
}

int main()
{
Test &&obj = GetTmp();
cout << "Resource from " << __func__ << ": " << (void *)obj.d << endl;

return 0;
}

和拷贝构造函数类似,有几点需要注意:

  • 参数(右值)的符号必须是右值引用符号,即“&&”。
  • 参数(右值)不可以是常量,因为我们需要修改右值。
  • 参数(右值)的资源链接和标记必须修改,否则,右值的析构函数就会释放资源,转移到新对象的资源也就无效了。

编译运行结果:

C++11:移动语义_转移构造函数_02

有了右值引用和转移语义,我们在设计和实现类时,对于需要动态申请大量资源的类,应该设计转移构造函数和转移赋值函数,以提高应用程序的效率。

转移赋值函数

#include <iostream>
using namespace std;

class Test
{
public:
Test(int a = 0)
{
d = new int(a);
cout << "构造函数\n";
}

Test(const Test & tmp)
{
d = new int;
*d = *(tmp.d);
cout << "拷贝构造函数\n";
}

Test & operator=(const Test & tmp)
{//赋值运算符重载函数
if(&tmp == this)
{
return *this;
}
delete d; //往往需要先把原来空间释放后,再申请,这里碰巧可以不这样做
d = new int;
*d = *(tmp.d);
cout << "赋值运算符重载函数\n";

return *this;
}

#if 1
Test(Test && tmp)
{ // 移动构造函数
d = tmp.d;
tmp.d = NULL; // 将临时值的指针成员置空
cout << "移动构造函数" << endl;
}

Test & operator=(Test && tmp)
{//转移赋值函数
if(&tmp == this)
{
return *this;
}

*d = *(tmp.d);
tmp.d = NULL;
cout << "转移赋值函数\n";

return *this;
}

#endif

~Test()
{
if(d != NULL)
{
delete d;
cout << "delete d\n";
}
cout << "析构函数\n";
}

int * d;
};

Test GetTmp()
{
Test h;
return h;
}

int main()
{
Test obj(10);
obj = GetTmp();//转移赋值函数

return 0;
}

编译运行结果:

C++11:移动语义_转移构造函数_03

参考资料:​​C++11 标准新特性: 右值引用与转移语义​


标签:11,tmp,cout,int,语义,C++,Test,转移,构造函数
From: https://blog.51cto.com/u_3002289/5720706

相关文章

  • C++11:std::move和std::forward
    标准库函数std::move既然编译器只对右值引用才能调用转移构造函数和转移赋值函数,而所有命名对象都只能是左值引用,如果已知一个命名对象不再被使用而想对它调用转移构造函数......
  • Java11安装(win11)
    下载JDK解压版本下载后解压,并放到一个没有中文路径的目录,如图所示:配置Java环境变量以Windows11系统为例,打开设置,搜索“环境”,点击“编辑系统环境变量”,如图所示:点击......
  • c++可执行文件产生过程
    https://blog.csdn.net/qq_34799070/article/details/125472381#ifndefTEST#defineTEST#include<iostream>usingnamespacestd;voidtest(){cout<<"h......
  • C++问题汇总——长期更新
    避免使用vector<bool>,尽量使用bitset或者deque<bool>因为vector<bool>不是容器,底层是bool值按bit存储STLlist容器数据结构上看是带空头的双向循环链表容器迭代......
  • 视频融合云平台EasyCVR级联时出现报错“Error 1146",是什么原因?
    EasyCVR具备强大的视频接入、汇聚与管理、视频分发、设备管理、用户及角色权限管理等能力。平台可提供的丰富的视频功能,包括:视频监控直播、云端录像、云存储、录像检索与回......
  • 【C++】关于智能指针的简单学习
    智能指针示例类:classString{private: stringm_value;public: String(stringstr):m_value(str){ cout<<"构造"<<m_value<<"\n"; } friendostream&o......
  • c++基本数据类型
    #include<iostream>#include<typeinfo>usingnamespacestd;intmain(){ inti; //定义一个整型变量 cout<<"输入一个整数i:"; cin>>i; cout<<"输出刚才......
  • 深圳工业智能网关BL110实现西门子S7-400 PLC 接入阿里云平台
    LAN接口的配置COM口采集西门子S7-400PLC的配置工业智能网关BL110一共有一个LAN接口,一个WAN接口,可以通过LAN接口采集数据,通过WAN接口接入局域网,设置过程不一样,WAN接口可......
  • 深圳工业智能网关BL110 实现西门子S7-400 PLC接入OPC UA云平台
    LAN接口的配置COM口采集西门子S7-400PLC的配置工业智能网关BL110一共有一个LAN接口,一个WAN接口,可以通过LAN接口采集数据,通过WAN接口接入局域网,设置过程不一样,WAN接口可......
  • 33、C++双目摄像头进行测距实验
    基本思想:因为最近用到了双目摄像头测距的代码逻辑,逐记录和转发一下大佬们的知识点,本菜鸡使用的深圳市鸿市康科技有限公司的双目摄像头进行测试 本测试需要使用pycharm和Mat......