首頁 > 軟體

C++ move semantic移動語意介紹

2022-08-25 10:01:20

前言

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

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內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com