C++ move semantic移動(dòng)語(yǔ)義介紹
前言
在說(shuō)移動(dòng)語(yǔ)義之前 本文作者假設(shè)你已經(jīng)具備了深拷貝淺拷貝左值右值等基本概念 本文不會(huì)再過(guò)多敘述 那么接下來(lái) 讓我們開始吧
Tips:(警告 警告 警告 警告)在閱讀本文章之前 作者首先提醒 線代編譯器有RVO和NRVO等一系列優(yōu)化策略 除非你明確知道你要使用std::move 不然我并不是很推薦你使用移動(dòng)語(yǔ)義 他很有可能是無(wú)意義的
移動(dòng)構(gòu)造
在說(shuō)移動(dòng)語(yǔ)義之前 讓我們先來(lái)說(shuō)說(shuō)移動(dòng)構(gòu)造這玩意
我們都知道 深拷貝是會(huì)把在堆區(qū)的內(nèi)存一起拷貝的 那么如果我們明確知道一個(gè)對(duì)象并不會(huì)再繼續(xù)使用 但是同時(shí)我們又想拿到他堆區(qū)的資源的時(shí)候 我們應(yīng)該怎么辦呢? 移動(dò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 << "移動(dòng)構(gòu)造函數(shù)調(diào)用" << std::endl;
this->p = tmp.p;
tmp.p = nullptr;
}
MoveClass& operator=(MoveClass&& tmp)
{
std::cout << "移動(dòng)構(gòu)造函數(shù)調(diào)用" << std::endl;
this->p = tmp.p;
tmp.p = nullptr;
}
};
MoveClass MoveClassTest(MoveClass d)
{
return MoveClass();
}
int main()
{
MoveClass cc;
//好 接下來(lái)我們不再想使用c了 但是堆區(qū)的資源我們并不想拷貝 那么使用如下構(gòu)造方式
MoveClass d(std::move(cc));
system("pause");
}移動(dòng)前數(shù)據(jù)如下圖所示:

移動(dòng)后數(shù)據(jù)如下圖所示:

程序輸出結(jié)果:

為什么我們需要move semantic
設(shè)想一個(gè)場(chǎng)景 我們?cè)谝粋€(gè)作用域申請(qǐng)了一個(gè)超級(jí)大的string 如下圖所示
#include <iostream>
#include <string.h>
void test1(std::string s)
{
std::cout << "test1:" << s.c_str()<<std::endl;;
}
void test()
{
std::string s = "超級(jí)大的string";
test1(s);
std::cout <<"test:"<< s.c_str() << std::endl;
return;
}
int main()
{
test();
system("pause");
}運(yùn)行結(jié)果如下:

你們就要說(shuō)了 有啥用啊 但是只要你懂一點(diǎn)c++ 你就會(huì)知道 在test中的s我們是不需要了的 也就是我們?cè)趖est是不想再繼續(xù)使用s的 但是在我們調(diào)用test1的時(shí)候 我們又重新拷貝了s一份 那么性能是不是就浪費(fèi)了呢?如果這個(gè)string超級(jí)大 你的程序是不是就很垃呢
我們只需要簡(jiǎn)簡(jiǎn)單單的加一個(gè)std::move 他就不是拷貝 而只是單純的移動(dòng)指針 如下
#include <iostream>
#include <string.h>
void test1(std::string s)
{
std::cout << "test1:" << s.c_str()<<std::endl;;
}
void test()
{
std::string s = "超級(jí)大的string";
test1(std::move(s));
std::cout <<"test:"<< s.c_str() << std::endl;
return;
}
int main()
{
test();
system("pause");
}運(yùn)行結(jié)果如下:

這就是他最最最本質(zhì)的作用 一個(gè)東西是左值時(shí) 你仍然想要他去觸發(fā)移動(dòng)構(gòu)造記住 其他時(shí)候你并不需要去考慮 因?yàn)榫幾g器有優(yōu)化懂嗎 不要嘗試自己去干擾編譯器的優(yōu)化 除非你真的非常非常非常清楚你自己正在干什么
到此這篇關(guān)于C++ move semantic移動(dòng)語(yǔ)義介紹的文章就介紹到這了,更多相關(guān)C++ move semantic內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
基于C++的農(nóng)夫過(guò)河問(wèn)題算法設(shè)計(jì)與實(shí)現(xiàn)方法
這篇文章主要介紹了基于C++的農(nóng)夫過(guò)河問(wèn)題算法設(shè)計(jì)與實(shí)現(xiàn)方法,簡(jiǎn)單描述了農(nóng)夫過(guò)河問(wèn)題,并結(jié)合實(shí)例形式詳細(xì)分析了基于C++實(shí)現(xiàn)農(nóng)夫過(guò)河問(wèn)題的相關(guān)算法實(shí)現(xiàn)步驟與操作技巧,需要的朋友可以參考下2017-09-09
VC++實(shí)現(xiàn)CStdioFile寫入及讀取文件并自動(dòng)換行的方法
這篇文章主要介紹了VC++實(shí)現(xiàn)CStdioFile寫入及讀取文件并自動(dòng)換行的方法,很實(shí)用的功能,需要的朋友可以參考下2014-08-08
win10環(huán)境下C++ vs2015編譯opencv249的教程
這篇文章主要介紹了win10環(huán)境下C++ vs2015編譯opencv249的教程,本文分步驟給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-03-03
C++?STL標(biāo)準(zhǔn)庫(kù)之std::list使用介紹及用法詳解
std::list是支持常數(shù)時(shí)間從容器任何位置插入和移除元素的容器,下面這篇文章主要給大家介紹了關(guān)于C++?STL標(biāo)準(zhǔn)庫(kù)之std::list使用介紹及用法詳解的相關(guān)資料,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-11-11
VSCode配置C/C++并添加非工作區(qū)頭文件的方法
這篇文章主要介紹了VSCode配置C/C++并添加非工作區(qū)頭文件的方法,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-03-03

