C++ move semantic移動語義介紹
前言
在說移動語義之前 本文作者假設(shè)你已經(jīng)具備了深拷貝淺拷貝左值右值等基本概念 本文不會再過多敘述 那么接下來 讓我們開始吧
Tips:(警告 警告 警告 警告)在閱讀本文章之前 作者首先提醒 線代編譯器有RVO和NRVO等一系列優(yōu)化策略 除非你明確知道你要使用std::move 不然我并不是很推薦你使用移動語義 他很有可能是無意義的
移動構(gòu)造
在說移動語義之前 讓我們先來說說移動構(gòu)造這玩意
我們都知道 深拷貝是會把在堆區(qū)的內(nèi)存一起拷貝的 那么如果我們明確知道一個對象并不會再繼續(xù)使用 但是同時(shí)我們又想拿到他堆區(qū)的資源的時(shí)候 我們應(yīng)該怎么辦呢? 移動構(gòu)造給我們提供了這種能力 代碼如下所示:
class MoveClass { public: int* p; MoveClass() { p = new int(); std::cout << "默認(rèn)構(gòu)造調(diào)用" << std::endl; } ~MoveClass() { std::cout << "析構(gòu)函數(shù)調(diào)用" << std::endl; if (!p) delete p; } MoveClass(MoveClass& tmp) { } MoveClass(MoveClass&& tmp) { std::cout << "移動構(gòu)造函數(shù)調(diào)用" << std::endl; this->p = tmp.p; tmp.p = nullptr; } MoveClass& operator=(MoveClass&& tmp) { std::cout << "移動構(gòu)造函數(shù)調(diào)用" << std::endl; this->p = tmp.p; tmp.p = nullptr; } }; MoveClass MoveClassTest(MoveClass d) { return MoveClass(); } int main() { MoveClass cc; //好 接下來我們不再想使用c了 但是堆區(qū)的資源我們并不想拷貝 那么使用如下構(gòu)造方式 MoveClass d(std::move(cc)); system("pause"); }
移動前數(shù)據(jù)如下圖所示:
移動后數(shù)據(jù)如下圖所示:
程序輸出結(jié)果:
為什么我們需要move semantic
設(shè)想一個場景 我們在一個作用域申請了一個超級大的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"); }
運(yùn)行結(jié)果如下:
你們就要說了 有啥用啊 但是只要你懂一點(diǎn)c++ 你就會知道 在test中的s我們是不需要了的 也就是我們在test是不想再繼續(xù)使用s的 但是在我們調(diào)用test1的時(shí)候 我們又重新拷貝了s一份 那么性能是不是就浪費(fèi)了呢?如果這個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"); }
運(yùn)行結(jié)果如下:
這就是他最最最本質(zhì)的作用 一個東西是左值時(shí) 你仍然想要他去觸發(fā)移動構(gòu)造記住 其他時(shí)候你并不需要去考慮 因?yàn)榫幾g器有優(yōu)化懂嗎 不要嘗試自己去干擾編譯器的優(yōu)化 除非你真的非常非常非常清楚你自己正在干什么
到此這篇關(guān)于C++ move semantic移動語義介紹的文章就介紹到這了,更多相關(guān)C++ move semantic內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
VC實(shí)現(xiàn)A進(jìn)程窗口嵌入到B進(jìn)程窗口中顯示的方法
這篇文章主要介紹了VC實(shí)現(xiàn)A進(jìn)程窗口嵌入到B進(jìn)程窗口中顯示的方法,對于理解windows程序運(yùn)行原理的進(jìn)程問題有一定的幫助,需要的朋友可以參考下2014-07-07解決c++?error:crosses?initialization?of?問題
最近在寫代碼的時(shí)候,碰到了?crosses?initialization?of?...?的問題,只因我在?switch?的某個?case?分支下定義了一個變量,于是乎便將這個問題整理一下,需要的朋友可以參考下2023-03-03使用C語言實(shí)現(xiàn)本地socke通訊的方法
這篇文章主要介紹了?使用C語言實(shí)現(xiàn)本地socke通訊,代碼分為服務(wù)器代碼和客戶端代碼,代碼簡單易懂,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-12-12Qt專欄之模態(tài)與非模態(tài)對話框的實(shí)現(xiàn)
這篇文章主要介紹了Qt專欄之模態(tài)與非模態(tài)對話框的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-04-04C語言輸出孿生素?cái)?shù)的實(shí)現(xiàn)示例
本文主要介紹了C語言輸出孿生素?cái)?shù)的實(shí)現(xiàn)示例,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-01-01C語言實(shí)現(xiàn)最小生成樹構(gòu)造算法
這篇文章主要為大家詳細(xì)介紹了C語言實(shí)現(xiàn)最小生成樹構(gòu)造算法,利用Prim算法或kruskal算法求解,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-01-01Embarcadero?Dev-C++輸出中文亂碼問題圖文詳解
Dev-C++(或者叫做 Dev-Cpp)是Windows環(huán)境下的一個輕量級C/C++ 集成開發(fā)環(huán)境(IDE),下面這篇文章主要給大家介紹了關(guān)于Embarcadero?Dev-C++輸出中文亂碼問題的相關(guān)資料,需要的朋友可以參考下2023-01-01