詳解C++11中的右值引用與移動(dòng)語義
C++11的一個(gè)最主要的特性就是可以移動(dòng)而非拷貝對(duì)象的能力。很多情況都會(huì)發(fā)生對(duì)象的拷貝,有時(shí)對(duì)象拷貝后就立即銷毀,在這些情況下,移動(dòng)而非拷貝對(duì)象會(huì)大幅度提升性能。
右值與右值引用
為了支持移動(dòng)操作,新標(biāo)準(zhǔn)引入了一種新的引用類型——右值引用,就是必須綁定到右值的引用。我們通過&&而不是&來獲得右值引用。右值引用一個(gè)重要的特性就是只能綁定到將要銷毀的對(duì)象。
左值和右值是表達(dá)式的屬性,一些表達(dá)式生成或要求左值,而另一些則生成或要求右值。一般而言,一個(gè)左值表達(dá)式表示的是一個(gè)對(duì)象的身份,而右值表達(dá)式表示的是對(duì)象的值。(可以取地址的、有名字的就是左值;不能取地址的、沒有名字的就是右值。)兩者明顯的區(qū)別就是左值有持久的狀態(tài),而右值要么是字面常量,要么是在表達(dá)式求值過程中創(chuàng)建的臨時(shí)對(duì)象。
類似于常規(guī)引用(左值引用),一個(gè)右值引用也不過是某個(gè)對(duì)象的另一個(gè)名字而已。我們不能將左值引用綁定到要求轉(zhuǎn)換的表達(dá)式、字面常量或是返回值的表達(dá)式,也不能把右值應(yīng)用直接綁定到一個(gè)左值上。但是,常量左值引用可以綁定到非常量左值、常量左值、右值,是一個(gè)萬能引用類型。不過相比右值引用所引用的右值,常量左值引用所引用的右值在它的“余生”中只能是只讀的。
int i = 42; int &r = i; //r引用i int &r2 = i*2; //錯(cuò)誤,i*2是一個(gè)右值 int &&rr = i; //錯(cuò)誤,不能將一個(gè)右值引用綁定到一個(gè)左值上 int &&rr2 = i*2; //正確,將rr2綁定到一個(gè)乘法結(jié)果上 const int &r3 = i*2; //正確,將一個(gè)常量引用綁定到一個(gè)右值上
變量可以看做只有一個(gè)運(yùn)算對(duì)象而沒有運(yùn)算符的表達(dá)式,是一個(gè)左值。我們不能將一個(gè)右值引用直接綁定到一個(gè)變量上,即使這個(gè)變量是右值引用類型。但是,我們可以通過新標(biāo)準(zhǔn)庫中的move函數(shù)來獲得綁定到左值上的右值引用。
int &&rr3 = std::move(rr2);
注意,被轉(zhuǎn)化的左值,其生命周期并沒有隨著左右至的轉(zhuǎn)化而改變,在轉(zhuǎn)換之后使用左值可能造成運(yùn)行時(shí)錯(cuò)誤。因此,調(diào)用move就意味著承諾:除了對(duì)原左值變量賦值或銷毀它外,我們將不再使用它。不過更多的時(shí)候,我們需要轉(zhuǎn)換成右值引用的還是一個(gè)確實(shí)生命周期即將結(jié)束的對(duì)象。
移動(dòng)構(gòu)造函數(shù)和移動(dòng)賦值運(yùn)算符
為了讓自定義類型也支持移動(dòng)操作,需要為其定義移動(dòng)構(gòu)造函數(shù)和移動(dòng)賦值運(yùn)算符。這兩個(gè)成員類似對(duì)應(yīng)的拷貝操作,但它們從給定對(duì)象竊取資源而不是拷貝資源。類似于拷貝構(gòu)造函數(shù),移動(dòng)構(gòu)造函數(shù)的第一個(gè)參數(shù)是該類類型的一個(gè)右值引用,任何額外的參數(shù)都必須有默認(rèn)實(shí)參。除了完成資源移動(dòng)外,移動(dòng)構(gòu)造函數(shù)還必須確保移后源對(duì)象處于有效的、可析構(gòu)的狀態(tài)。
#include <iostream> #include <algorithm> class MemoryBlock { public: // 構(gòu)造函數(shù) explicit MemoryBlock(size_t length) : _length(length) , _data(new int[length]) {} // 析構(gòu)函數(shù) ~MemoryBlock() { if (_data != nullptr) delete[] _data; } // 拷貝賦值運(yùn)算符 MemoryBlock& operator=(const MemoryBlock& other) { if (this != &other) { delete[] _data; _length = other._length; _data = new int[_length]; std::copy(other._data, other._data + _length, _data); } return *this; } // 拷貝構(gòu)造函數(shù) MemoryBlock(const MemoryBlock& other) : _length(0) , _data(nullptr) { *this = other; } // 移動(dòng)賦值運(yùn)算符,通知標(biāo)準(zhǔn)庫該構(gòu)造函數(shù)不拋出任何異常(如果拋出異常會(huì)怎么樣?) MemoryBlock& operator=(MemoryBlock&& other) noexcept { if (this != &other) { delete[] _data; // 移動(dòng)資源 _data = other._data; _length = other._length; // 使移后源對(duì)象處于可銷毀狀態(tài) other._data = nullptr; other._length = 0; } return *this; } // 移動(dòng)構(gòu)造函數(shù) MemoryBlock(MemoryBlock&& other) noexcept _data(nullptr) , _length(0) { *this = std::move(other); } size_t Length() const { return _length; } private: size_t _length; // The length of the resource. int* _data; // The resource. };
只有當(dāng)一個(gè)類沒有定義任何自己版本的拷貝控制成員,且類的每個(gè)非static數(shù)據(jù)成員都可移動(dòng)時(shí),編譯器才會(huì)為它合成移動(dòng)構(gòu)造函數(shù)會(huì)移動(dòng)賦值運(yùn)算符。編譯器可以移動(dòng)內(nèi)置類型;如果一個(gè)類類型有對(duì)應(yīng)的移動(dòng)操作,編譯器也能移動(dòng)這個(gè)類型的成員。此外,定義了一個(gè)移動(dòng)構(gòu)造函數(shù)或移動(dòng)賦值運(yùn)算符的類必須也定義自己的拷貝操作;否則,這些成員默認(rèn)地定義為刪除的。而移動(dòng)操作則不同,它永遠(yuǎn)不會(huì)隱式定義為刪除的。但如果我們顯式地要求編譯器生成=defualt的移動(dòng)操作,且編譯器不能移動(dòng)所有成員,則編譯器會(huì)將移動(dòng)操作定義為刪除的函數(shù)。
如果一個(gè)類既有移動(dòng)構(gòu)造函數(shù)又有拷貝構(gòu)造函數(shù),編譯會(huì)使用普通的函數(shù)匹配規(guī)則來確定使用哪個(gè)構(gòu)造函數(shù)。但如果只定義了拷貝操作而未定義移動(dòng)操作,編譯器不會(huì)合成移動(dòng)構(gòu)造函數(shù),此時(shí)即使調(diào)用move來移動(dòng)它們,也是調(diào)用的拷貝操作。
class Foo{ public: Foo() = default; Foo(const Foo&); // 為定義移動(dòng)構(gòu)造函數(shù) }; Foo x; Foo y(x); //調(diào)用拷貝構(gòu)造函數(shù) Foo z(std::move(x)); //調(diào)用拷貝構(gòu)造函數(shù),因?yàn)槲炊x移動(dòng)構(gòu)造函數(shù)
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持我們。
您可能感興趣的文章
- 01-10求子數(shù)組最大和的解決方法詳解
- 01-10深入二叉樹兩個(gè)結(jié)點(diǎn)的最低共同父結(jié)點(diǎn)的詳解
- 01-10數(shù)據(jù)結(jié)構(gòu)課程設(shè)計(jì)- 解析最少換車次數(shù)的問題詳解
- 01-10數(shù)據(jù)結(jié)構(gòu)課程設(shè)計(jì)-用棧實(shí)現(xiàn)表達(dá)式求值的方法詳解
- 01-10HDOJ 1443 約瑟夫環(huán)的最新應(yīng)用分析詳解
- 01-10使用C++實(shí)現(xiàn)全排列算法的方法詳解
- 01-10如何查看進(jìn)程實(shí)際的內(nèi)存占用情況詳解
- 01-10深入Main函數(shù)中的參數(shù)argc,argv的使用詳解
- 01-10APUE筆記之:進(jìn)程環(huán)境詳解
- 01-10深入第K大數(shù)問題以及算法概要的詳解


閱讀排行
- 1C語言 while語句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹的示例代碼(圣誕
- 3利用C語言實(shí)現(xiàn)“百馬百擔(dān)”問題方法
- 4C語言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言的正則匹配函數(shù) c語言正則表達(dá)
- 04-02c語言用函數(shù)寫分段 用c語言表示分段
- 04-02c語言中對(duì)數(shù)函數(shù)的表達(dá)式 c語言中對(duì)
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段
- 04-02C語言中怎么打出三角函數(shù) c語言中怎
- 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求
隨機(jī)閱讀
- 01-10使用C語言求解撲克牌的順子及n個(gè)骰子
- 01-11ajax實(shí)現(xiàn)頁面的局部加載
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 04-02jquery與jsp,用jquery
- 01-10C#中split用法實(shí)例總結(jié)
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 01-10delphi制作wav文件的方法