中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

如何進行C++ 11右值引用的理解

發布時間:2021-10-29 10:25:01 來源:億速云 閱讀:101 作者:柒染 欄目:編程語言

本篇文章給大家分享的是有關如何進行C++ 11右值引用的理解,小編覺得挺實用的,因此分享給大家學習,希望大家閱讀完這篇文章后可以有所收獲,話不多說,跟著小編一起來看看吧。

C++ 11中引入的一個非常重要的概念就是右值引用。理解右值引用是學習“移動語義”(move semantics)的基礎。而要理解右值引用,就必須先區分左值與右值。

對左值和右值的一個最常見的誤解是:等號左邊的就是左值,等號右邊的就是右值。左值和右值都是針對表達式而言的,左值是指表達式結束后依然存在的持久對象,右值是指表達式結束時就不再存在的臨時對象。一個區分左值與右值的便捷方法是:看能不能對表達式取地址,如果能,則為左值,否則為右值。下面給出一些例子來進行說明。

int a = 10

int b = 20

int *pFlag = &a;

vector<int> vctTemp;

vctTemp.push_back(1);

string str1 = "hello "

string str2 = "world"

const int &m = 1

請問,a,b, a+b, a++, ++a, pFlag, *pFlag, vctTemp[0], 100, string("hello"), str1, str1+str2, m分別是左值還是右值?

a和b都是持久對象(可以對其取地址),是左值;

a+b是臨時對象(不可以對其取地址),是右值;

a++是先取出持久對象a的一份拷貝,再使持久對象a的值加1,***返回那份拷貝,而那份拷貝是臨時對象(不可以對其取地址),故其是右值;

++a則是使持久對象a的值加1,并返回那個持久對象a本身(可以對其取地址),故其是左值;

pFlag和*pFlag都是持久對象(可以對其取地址),是左值;

vctTemp[0]調用了重載的[]操作符,而[]操作符返回的是一個int &,為持久對象(可以對其取地址),是左值;

100和string("hello")是臨時對象(不可以對其取地址),是右值;

str1是持久對象(可以對其取地址),是左值;

str1+str2是調用了+操作符,而+操作符返回的是一個string(不可以對其取地址),故其為右值;

m是一個常量引用,引用到一個右值,但引用本身是一個持久對象(可以對其取地址),為左值。

區分清楚了左值與右值,我們再來看看左值引用。左值引用根據其修飾符的不同,可以分為非常量左值引用和常量左值引用。

非常量左值引用只能綁定到非常量左值,不能綁定到常量左值、非常量右值和常量右值。如果允許綁定到常量左值和常量右值,則非常量左值引用可以用于修改常量左值和常量右值,這明顯違反了其常量的含義。如果允許綁定到非常量右值,則會導致非常危險的情況出現,因為非常量右值是一個臨時對象,非常量左值引用可能會使用一個已經被銷毀了的臨時對象。

常量左值引用可以綁定到所有類型的值,包括非常量左值、常量左值、非常量右值和常量右值。

可以看出,使用左值引用時,我們無法區分出綁定的是否是非常量右值的情況。那么,為什么要對非常量右值進行區分呢,區分出來了又有什么好處呢?這就牽涉到C++中一個著名的性能問題&mdash;&mdash;拷貝臨時對象。考慮下面的代碼:

vector<int> GetAllScores()

{

vector<int> vctTemp;

vctTemp.push_back(90);

vctTemp.push_back(95);

return vctTemp;

}

當使用vector<int> vctScore = GetAllScores()進行初始化時,實際上調用了三次構造函數。盡管有些編譯器可以采用RVO(Return Value Optimization)來進行優化,但優化工作只在某些特定條件下才能進行。可以看到,上面很普通的一個函數調用,由于存在臨時對象的拷貝,導致了額外的兩次拷貝構造函數和析構函數的開銷。當然,我們也可以修改函數的形式為void GetAllScores(vector<int> &vctScore),但這并不一定就是我們需要的形式。另外,考慮下面字符串的連接操作:

string s1("hello");

string s = s1 + "a" + "b" + "c" + "d" + "e"

在對s進行初始化時,會產生大量的臨時對象,并涉及到大量字符串的拷貝操作,這顯然會影響程序的效率和性能。怎么解決這個問題呢?如果我們能確定某個值是一個非常量右值(或者是一個以后不會再使用的左值),則我們在進行臨時對象的拷貝時,可以不用拷貝實際的數據,而只是“竊取”指向實際數據的指針(類似于STL中的auto_ptr,會轉移所有權)。C++ 11中引入的右值引用正好可用于標識一個非常量右值。C++ 11中用&表示左值引用,用&&表示右值引用,如:

int &&a = 10

右值引用根據其修飾符的不同,也可以分為非常量右值引用和常量右值引用。

非常量右值引用只能綁定到非常量右值,不能綁定到非常量左值、常量左值和常量右值(VS2010 beta版中可以綁定到非常量左值和常量左值,但正式版中為了安全起見,已不允許)。如果允許綁定到非常量左值,則可能會錯誤地竊取一個持久對象的數據,而這是非常危險的;如果允許綁定到常量左值和常量右值,則非常量右值引用可以用于修改常量左值和常量右值,這明顯違反了其常量的含義。

常量右值引用可以綁定到非常量右值和常量右值,不能綁定到非常量左值和常量左值(理由同上)。

有了右值引用的概念,我們就可以用它來實現下面的CMyString類。

class CMyString

{

public:

// 構造函數

CMyString(const char *pszSrc = NULL)

{

cout << "CMyString(const char *pszSrc = NULL)" << endl;

if (pszSrc == NULL)

{

m_pData = new char[1];

*m_pData = '

以上就是如何進行C++ 11右值引用的理解,小編相信有部分知識點可能是我們日常工作會見到或用到的。希望你能通過這篇文章學到更多知識。更多詳情敬請關注億速云行業資訊頻道。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

c++
AI

奉新县| 邵阳县| 永胜县| 罗定市| 乌什县| 光泽县| 仪陇县| 吴旗县| 高雄市| 琼中| 吉安市| 文登市| 中山市| 寿宁县| 杨浦区| 德惠市| 佛学| 西乡县| 铜梁县| 上虞市| 中西区| 明溪县| 萨嘎县| 乐东| 阿拉尔市| 隆林| 于田县| 吴忠市| 建水县| 咸阳市| 蒙自县| 吉隆县| 高安市| 东城区| 喜德县| 林州市| 无锡市| 合作市| 崇义县| 平利县| 克什克腾旗|