c++右值引用之转移构造函数

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

在现有的 C++ 机制中,我们可以定义拷贝构造函数和赋值函数。要实现转移语义,需要定义转移构造函数,还可以定义转移赋值操作符。对于右值的拷贝和赋值会调用转移构造函数和转移赋值操作符。如果转移构造函数和转移拷贝操作符没有定义,那么就遵循现有的机制,拷贝构造函数和赋值操作符会被调用。
这里有关于拷贝构造函数与移动构造函数的push_back()使用c++之emplace_back()与push_back()区别

下面就用示例演示一下,参考C++ 11 左值,右值,左值引用,右值引用,std::move, std::foward

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

using namespace std;

class MyString {
private:
	char* _data;
	size_t   _len;
	void _init_data(const char *s) {
		_data = new char[_len + 1];
		memcpy(_data, s, _len);
		_data[_len] = '\0';
	}
public:
//构造函数
	MyString() {
		_data = NULL;
		_len = 0;
	}
//重载构造函数
	MyString(const char* p) {
		_len = strlen(p);
		_init_data(p);
	}
//拷贝构造函数
	MyString(const MyString& str) {
		_len = str._len;
		_init_data(str._data);
		std::cout << "Copy Constructor is called! source: " << str._data << std::endl;
	}
//拷贝赋值操作符
	MyString& operator=(const MyString& str) {
		if (this != &str) {
			_len = str._len;
			_init_data(str._data);
		}
		std::cout << "Copy Assignment is called! source: " << str._data << std::endl;
		return *this;
	}
//移动构造函数
	MyString(MyString&& str) {
		std::cout << "Move Constructor is called! source: " << str._data << std::endl;
		_len = str._len;
		_data = str._data;
		str._len = 0;
		str._data = NULL;
	}
	//移动赋值运算符
	MyString& operator=(MyString&& str) {
		std::cout << "Move Assignment is called! source: " << str._data << std::endl;
		if (this != &str) {
			_len = str._len;
			_data = str._data;
			str._len = 0;
			str._data = NULL;
		}
		return *this;
	}


	virtual ~MyString() {
		if (_data) free(_data);
	}
};

int main() {
	MyString a;
	a = MyString("Hello");//调用赋值运算符
	MyString b = a;//只调用拷贝构造函数
	cout << "----------------"<<endl;
	std::vector<MyString> vec;
	vec.push_back(MyString("World"));

	system("pause");
}

只存在拷贝构造函数和拷贝赋值操作符结果(注释掉移动构造函数和移动赋值操作符)
在这里插入图片描述
这个 string 类已经基本满足我们演示的需要。在 main 函数中,实现了调用拷贝构造函数的操作和拷贝赋值操作符的操作。MyString(“Hello”) 和 MyString(“World”) 都是临时对象,也就是右值。虽然它们是临时的,但程序仍然调用了拷贝构造和拷贝赋值,造成了没有意义的资源申请和释放的操作。如果能够直接使用临时对象已经申请的资源,既能节省资源,有能节省资源申请和释放的时间。这正是定义转移语义的目的

存在移动构造函数和移动赋值操作符结果
在这里插入图片描述
关于转移构造函数有下面几点需要对照代码注意:

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

原因:
我们可以看一下析构函数

virtual ~MyString() {
		if (_data) free(_data);
	}

它会delete掉_data的内存空间。但是如果调用析构函数的时候_data指向的是NULL,那么就不会delte任何内存空间。
所以如果转移构造函数或者移动赋值操作符中没有

str._data = NULL;

虽然调用移动构造函数或者移动赋值操作符后,已经获得了右值的_data的内存空间,但是之后右值就被销毁了,那么获得的的那片内存也被释放了,指向的就是一个不合法的内存空间。所以我们就要防止这片空间被销毁。

  • 3
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值