日本免费高清视频-国产福利视频导航-黄色在线播放国产-天天操天天操天天操天天操|www.shdianci.com

學(xué)無(wú)先后,達(dá)者為師

網(wǎng)站首頁(yè) 編程語(yǔ)言 正文

C++?move?semantic移動(dòng)語(yǔ)義介紹_C 語(yǔ)言

作者:殺神李 ? 更新時(shí)間: 2022-10-21 編程語(yǔ)言

前言

在說(shuō)移動(dòng)語(yǔ)義之前 本文作者假設(shè)你已經(jīng)具備了深拷貝淺拷貝左值右值等基本概念 本文不會(huì)再過(guò)多敘述 那么接下來(lái) 讓我們開(kāi)始吧

Tips:(警告 警告 警告 警告)在閱讀本文章之前 作者首先提醒 線(xiàn)代編譯器有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)化 除非你真的非常非常非常清楚你自己正在干什么

原文鏈接:https://blog.csdn.net/qq_16401691/article/details/126424125

欄目分類(lèi)
最近更新