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!
推薦閱讀:
- C++移動語義詳細介紹使用
- C++右值引用與move和forward函數的使用詳解
- C++ 內存分區模型的使用(代碼區、全局區、棧區、堆區、new)
- C++右值引用與移動構造函數基礎與應用詳解
- C++模板編程特性之移動語義