C++ move semantic移動語義介紹

前言

在說移動語義之前 本文作者假設你已經具備瞭深拷貝淺拷貝左值右值等基本概念 本文不會再過多敘述 那麼接下來 讓我們開始吧

Tips:(警告 警告 警告 警告)在閱讀本文章之前 作者首先提醒 線代編譯器有RVO和NRVO等一系列優化策略 除非你明確知道你要使用std::move 不然我並不是很推薦你使用移動語義 他很有可能是無意義的

移動構造

在說移動語義之前 讓我們先來說說移動構造這玩意

我們都知道 深拷貝是會把在堆區的內存一起拷貝的 那麼如果我們明確知道一個對象並不會再繼續使用 但是同時我們又想拿到他堆區的資源的時候 我們應該怎麼辦呢? 移動構造給我們提供瞭這種能力 代碼如下所示:

class MoveClass
{
public:
	int* p;
	MoveClass()
	{
		p = new int();
		std::cout << "默認構造調用" << std::endl;
	}
	~MoveClass()
	{
		std::cout << "析構函數調用" << std::endl;
		if (!p)
			delete p;
	}
	MoveClass(MoveClass& tmp)
	{
 
	}
	 MoveClass(MoveClass&& tmp)
	{
		std::cout << "移動構造函數調用" << std::endl;
		this->p = tmp.p;
		tmp.p = nullptr;
	}
	MoveClass& operator=(MoveClass&& tmp)
	 {
		 std::cout << "移動構造函數調用" << std::endl;
		 this->p = tmp.p;
		 tmp.p = nullptr;
	 }
};
MoveClass MoveClassTest(MoveClass d)
{
	return MoveClass();
}
int main()
{
	MoveClass cc;
	//好 接下來我們不再想使用c瞭 但是堆區的資源我們並不想拷貝 那麼使用如下構造方式
	MoveClass d(std::move(cc));
	system("pause");
}

移動前數據如下圖所示:

移動後數據如下圖所示:

程序輸出結果:

為什麼我們需要move semantic

設想一個場景 我們在一個作用域申請瞭一個超級大的string 如下圖所示

#include <iostream>
#include <string.h>
void test1(std::string s)
{
	std::cout << "test1:" << s.c_str()<<std::endl;;
}
void test()
{
	std::string s = "超級大的string";
	test1(s);
	std::cout <<"test:"<< s.c_str() << std::endl;
	return;
}
int main()
{
	test();
	system("pause");
}

運行結果如下:

你們就要說瞭 有啥用啊 但是隻要你懂一點c++ 你就會知道 在test中的s我們是不需要瞭的 也就是我們在test是不想再繼續使用s的 但是在我們調用test1的時候 我們又重新拷貝瞭s一份 那麼性能是不是就浪費瞭呢?如果這個string超級大 你的程序是不是就很垃呢

我們隻需要簡簡單單的加一個std::move 他就不是拷貝 而隻是單純的移動指針 如下

#include <iostream>
#include <string.h>
void test1(std::string s)
{
	std::cout << "test1:" << s.c_str()<<std::endl;;
}
void test()
{
	std::string s = "超級大的string";
	test1(std::move(s));
	std::cout <<"test:"<< s.c_str() << std::endl;
	return;
}
int main()
{
	test();
	system("pause");
}

運行結果如下:

這就是他最最最本質的作用 一個東西是左值時 你仍然想要他去觸發移動構造記住 其他時候你並不需要去考慮 因為編譯器有優化懂嗎 不要嘗試自己去幹擾編譯器的優化 除非你真的非常非常非常清楚你自己正在幹什麼

到此這篇關於C++ move semantic移動語義介紹的文章就介紹到這瞭,更多相關C++ move semantic內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: