自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

左值引用、右值引用、移動(dòng)語(yǔ)義、完美轉(zhuǎn)發(fā),你知道的不知道的都在這里

開(kāi)發(fā) 后端
眾所周知C++11新增了右值引用,談?dòng)抑狄梦覀円部梢詳U(kuò)展一些相關(guān)概念,程序喵下面會(huì)一一介紹。

 [[337215]]

眾所周知C++11新增了右值引用,談?dòng)抑狄梦覀円部梢詳U(kuò)展一些相關(guān)概念:

  •  左值
  •  右值
  •  純右值
  •  將亡值
  •  左值引用
  •  右值引用
  •  移動(dòng)語(yǔ)義
  •  完美轉(zhuǎn)發(fā)
  •  返回值優(yōu)化

程序喵下面會(huì)一一介紹:

左值、右值

概念1:

左值:可以放到等號(hào)左邊的東西叫左值。

右值:不可以放到等號(hào)左邊的東西就叫右值。

概念2:

左值:可以取地址并且有名字的東西就是左值。

右值:不能取地址的沒(méi)有名字的東西就是右值。

舉例: 

  1. int a = b + c; 

a是左值,有變量名,可以取地址,也可以放到等號(hào)左邊, 表達(dá)式b+c的返回值是右值,沒(méi)有名字且不能取地址,&(b+c)不能通過(guò)編譯,而且也不能放到等號(hào)左邊。 

  1. int a = 4; // a是左值,4作為普通字面量是右值 

左值一般有:

  •  函數(shù)名和變量名
  •  返回左值引用的函數(shù)調(diào)用
  •  前置自增自減表達(dá)式++i、--i
  •  由賦值表達(dá)式或賦值運(yùn)算符連接的表達(dá)式(a=b, a += b等)
  •  解引用表達(dá)式*p
  •  字符串字面值"abcd"

純右值、將亡值

純右值和將亡值都屬于右值。

純右值

運(yùn)算表達(dá)式產(chǎn)生的臨時(shí)變量、不和對(duì)象關(guān)聯(lián)的原始字面量、非引用返回的臨時(shí)變量、lambda表達(dá)式等都是純右值。

舉例:

  •  除字符串字面值外的字面值
  •  返回非引用類(lèi)型的函數(shù)調(diào)用
  •  后置自增自減表達(dá)式i++、i--
  •  算術(shù)表達(dá)式(a+b, a*b, a&&b, a==b等)
  •  取地址表達(dá)式等(&a)

將亡值

將亡值是指C++11新增的和右值引用相關(guān)的表達(dá)式,通常指將要被移動(dòng)的對(duì)象、T&&函數(shù)的返回值、std::move函數(shù)的返回值、轉(zhuǎn)換為T(mén)&&類(lèi)型轉(zhuǎn)換函數(shù)的返回值,將亡值可以理解為即將要銷(xiāo)毀的值,通過(guò)“盜取”其它變量?jī)?nèi)存空間方式獲取的值,在確保其它變量不再被使用或者即將被銷(xiāo)毀時(shí),可以避免內(nèi)存空間的釋放和分配,延長(zhǎng)變量值的生命周期,常用來(lái)完成移動(dòng)構(gòu)造或者移動(dòng)賦值的特殊任務(wù)。

舉例: 

  1. class A {  
  2.     xxx;  
  3. };  
  4. A a;  
  5. auto c = std::move(a); // c是將亡值  
  6. auto d = static_cast<A&&>(a); // d是將亡值 

左值引用、右值引用

根據(jù)名字大概就可以猜到意思,左值引用就是對(duì)左值進(jìn)行引用的類(lèi)型,右值引用就是對(duì)右值進(jìn)行引用的類(lèi)型,他們都是引用,都是對(duì)象的一個(gè)別名,并不擁有所綁定對(duì)象的堆存,所以都必須立即初始化。 

  1. type &name = exp; // 左值引用  
  2. type &&name = exp; // 右值引用 

左值引用

看代碼: 

  1. int a = 5 
  2. int &b = a; // b是左值引用  
  3. b = 4 
  4. int &c = 10; // error,10無(wú)法取地址,無(wú)法進(jìn)行引用  
  5. const int &d = 10; // ok,因?yàn)槭浅R?,引用常量?shù)字,這個(gè)常量數(shù)字會(huì)存儲(chǔ)在內(nèi)存中,可以取地址 

可以得出結(jié)論:對(duì)于左值引用,等號(hào)右邊的值必須可以取地址,如果不能取地址,則會(huì)編譯失敗,或者可以使用const引用形式,但這樣就只能通過(guò)引用來(lái)讀取輸出,不能修改數(shù)組,因?yàn)槭浅A恳谩?/p>

右值引用

如果使用右值引用,那表達(dá)式等號(hào)右邊的值需要時(shí)右值,可以使用std::move函數(shù)強(qiáng)制把左值轉(zhuǎn)換為右值。 

  1. int a = 4 
  2. int &&b = a; // error, a是左值  
  3. int &&c = std::move(a); // ok 

移動(dòng)語(yǔ)義

談移動(dòng)語(yǔ)義前,我們首先需要了解深拷貝與淺拷貝的概念

深拷貝、淺拷貝

直接拿代碼舉例: 

  1. class A {  
  2. public:  
  3.     A(int size) : size_(size) {  
  4.         data_ = new int[size];  
  5.     }  
  6.     A(){}  
  7.     A(const A& a) {  
  8.         size_ = a.size_;  
  9.         data_ = a.data_;  
  10.         cout << "copy " << endl 
  11.     }  
  12.     ~A() {  
  13.         delete[] data_;  
  14.     }  
  15.     int *data_;  
  16.     int size_;  
  17. };  
  18. int main() {  
  19.     A a(10);  
  20.     A b = a 
  21.     cout << "b " << b.data_ << endl 
  22.     cout << "a " << a.data_ << endl 
  23.     return 0;  

上面代碼中,兩個(gè)輸出的是相同的地址,a和b的data_指針指向了同一塊內(nèi)存,這就是淺拷貝,只是數(shù)據(jù)的簡(jiǎn)單賦值,那再析構(gòu)時(shí)data_內(nèi)存會(huì)被釋放兩次,導(dǎo)致程序出問(wèn)題,這里正常會(huì)出現(xiàn)double free導(dǎo)致程序崩潰的,但是不知道為什么我自己測(cè)試程序卻沒(méi)有崩潰,能力有限,沒(méi)搞明白,無(wú)論怎樣,這樣的程序肯定是有隱患的,如何消除這種隱患呢,可以使用如下深拷貝: 

  1. class A {  
  2. public:  
  3.     A(int size) : size_(size) {  
  4.         data_ = new int[size];  
  5.     }  
  6.     A(){}  
  7.     A(const A& a) {  
  8.         size_ = a.size_;  
  9.         data_ = new int[size_]; 
  10.         cout << "copy " << endl 
  11.     }  
  12.     ~A() {  
  13.         delete[] data_;  
  14.     }  
  15.     int *data_;  
  16.     int size_;  
  17. };  
  18. int main() {  
  19.     A a(10);  
  20.     A b = a 
  21.     cout << "b " << b.data_ << endl 
  22.     cout << "a " << a.data_ << endl 
  23.     return 0;  

深拷貝就是再拷貝對(duì)象時(shí),如果被拷貝對(duì)象內(nèi)部還有指針引用指向其它資源,自己需要重新開(kāi)辟一塊新內(nèi)存存儲(chǔ)資源,而不是簡(jiǎn)單的賦值。

聊完了深拷貝淺拷貝,可以聊聊移動(dòng)語(yǔ)義啦:

移動(dòng)語(yǔ)義,在程序喵看來(lái)可以理解為轉(zhuǎn)移所有權(quán),之前的拷貝是對(duì)于別人的資源,自己重新分配一塊內(nèi)存存儲(chǔ)復(fù)制過(guò)來(lái)的資源,而對(duì)于移動(dòng)語(yǔ)義,類(lèi)似于轉(zhuǎn)讓或者資源竊取的意思,對(duì)于那塊資源,轉(zhuǎn)為自己所擁有,別人不再擁有也不會(huì)再使用,通過(guò)C++11新增的移動(dòng)語(yǔ)義可以省去很多拷貝負(fù)擔(dān),怎么利用移動(dòng)語(yǔ)義呢,是通過(guò)移動(dòng)構(gòu)造函數(shù)。 

  1. class A {  
  2. public:  
  3.     A(int size) : size_(size) {  
  4.         data_ = new int[size];  
  5.     }  
  6.     A(){}  
  7.     A(const A& a) {  
  8.         size_ = a.size_;  
  9.         data_ = new int[size_];  
  10.         cout << "copy " << endl 
  11.     }  
  12.     A(A&& a) {  
  13.         this->data_ = a.data_; 
  14.          a.data_ = nullptr 
  15.         cout << "move " << endl 
  16.     }  
  17.     ~A() {  
  18.         if (data_ != nullptr) {  
  19.          delete[] data_;  
  20.         }  
  21.     }  
  22.     int *data_;  
  23.     int size_; 
  24.  };  
  25. int main() {  
  26.     A a(10);  
  27.     A b = a 
  28.     A c = std::move(a); // 調(diào)用移動(dòng)構(gòu)造函數(shù)  
  29.     return 0;  

如果不使用std::move(),會(huì)有很大的拷貝代價(jià),使用移動(dòng)語(yǔ)義可以避免很多無(wú)用的拷貝,提供程序性能,C++所有的STL都實(shí)現(xiàn)了移動(dòng)語(yǔ)義,方便我們使用。例如: 

  1. std::vector<string> vecs;  
  2. ...  
  3. std::vector<string> vecm = std::move(vecs); // 免去很多拷貝 

注意:移動(dòng)語(yǔ)義僅針對(duì)于那些實(shí)現(xiàn)了移動(dòng)構(gòu)造函數(shù)的類(lèi)的對(duì)象,對(duì)于那種基本類(lèi)型int、float等沒(méi)有任何優(yōu)化作用,還是會(huì)拷貝,因?yàn)樗鼈儗?shí)現(xiàn)沒(méi)有對(duì)應(yīng)的移動(dòng)構(gòu)造函數(shù)。

完美轉(zhuǎn)發(fā)

完美轉(zhuǎn)發(fā)指可以寫(xiě)一個(gè)接受任意實(shí)參的函數(shù)模板,并轉(zhuǎn)發(fā)到其它函數(shù),目標(biāo)函數(shù)會(huì)收到與轉(zhuǎn)發(fā)函數(shù)完全相同的實(shí)參,轉(zhuǎn)發(fā)函數(shù)實(shí)參是左值那目標(biāo)函數(shù)實(shí)參也是左值,轉(zhuǎn)發(fā)函數(shù)實(shí)參是右值那目標(biāo)函數(shù)實(shí)參也是右值。那如何實(shí)現(xiàn)完美轉(zhuǎn)發(fā)呢,答案是使用std::forward()。 

  1. void PrintV(int &t) {  
  2.     cout << "lvalue" << endl 
  3.  
  4. void PrintV(int &&t) {  
  5.     cout << "rvalue" << endl 
  6.  
  7. template<typename T>  
  8. void Test(T &&t) {  
  9.     PrintV(t);  
  10.     PrintV(std::forward<T>(t));   
  11.     PrintV(std::move(t));  
  12.  
  13. int main() {  
  14.     Test(1); // lvalue rvalue rvalue  
  15.     int a = 1 
  16.     Test(a); // lvalue lvalue rvalue  
  17.     Test(std::forward<int>(a)); // lvalue rvalue rvalue  
  18.     Test(std::forward<int&>(a)); // lvalue lvalue rvalue  
  19.     Test(std::forward<int&&>(a)); // lvalue rvalue rvalue  
  20.     return 0;  

分析

  •  Test(1):1是右值,模板中T &&t這種為萬(wàn)能引用,右值1傳到Test函數(shù)中變成了右值引用,但是調(diào)用PrintV()時(shí)候,t變成了左值,因?yàn)樗兂闪艘粋€(gè)擁有名字的變量,所以打印lvalue,而PrintV(std::forward<T>(t))時(shí)候,會(huì)進(jìn)行完美轉(zhuǎn)發(fā),按照原來(lái)的類(lèi)型轉(zhuǎn)發(fā),所以打印rvalue,PrintV(std::move(t))毫無(wú)疑問(wèn)會(huì)打印rvalue。
  •  Test(a):a是左值,模板中T &&這種為萬(wàn)能引用,左值a傳到Test函數(shù)中變成了左值引用,所以有代碼中打印。
  •  Test(std::forward<T>(a)):轉(zhuǎn)發(fā)為左值還是右值,依賴(lài)于T,T是左值那就轉(zhuǎn)發(fā)為左值,T是右值那就轉(zhuǎn)發(fā)為右值。

返回值優(yōu)化

返回值優(yōu)化(RVO)是一種C++編譯優(yōu)化技術(shù),當(dāng)函數(shù)需要返回一個(gè)對(duì)象實(shí)例時(shí)候,就會(huì)創(chuàng)建一個(gè)臨時(shí)對(duì)象并通過(guò)復(fù)制構(gòu)造函數(shù)將目標(biāo)對(duì)象復(fù)制到臨時(shí)對(duì)象,這里有復(fù)制構(gòu)造函數(shù)和析構(gòu)函數(shù)會(huì)被多余的調(diào)用到,有代價(jià),而通過(guò)返回值優(yōu)化,C++標(biāo)準(zhǔn)允許省略調(diào)用這些復(fù)制構(gòu)造函數(shù)。

那什么時(shí)候編譯器會(huì)進(jìn)行返回值優(yōu)化呢?

  •  return的值類(lèi)型與函數(shù)的返回值類(lèi)型相同
  •  return的是一個(gè)局部對(duì)象

看幾個(gè)例子:

示例1: 

  1. std::vector<int> return_vector(void) {  
  2.     std::vector<int> tmp {1,2,3,4,5};  
  3.     return tmp;  
  4.  
  5. std::vector<int> &&rval_ref = return_vector(); 

不會(huì)觸發(fā)RVO,拷貝構(gòu)造了一個(gè)臨時(shí)的對(duì)象,臨時(shí)對(duì)象的生命周期和rval_ref綁定,等價(jià)于下面這段代碼: 

  1. const std::vector<int>rval_ref = return_vector(); 

示例2: 

  1. std::vector<int>&& return_vector(void) {  
  2.     std::vector<int> tmp {1,2,3,4,5};  
  3.     return std::move(tmp); 
  4.  
  5. std::vector<int> &&rval_ref = return_vector(); 

這段代碼會(huì)造成運(yùn)行時(shí)錯(cuò)誤,因?yàn)閞val_ref引用了被析構(gòu)的tmp。講道理來(lái)說(shuō)這段代碼是錯(cuò)的,但我自己運(yùn)行過(guò)程中卻成功了,我沒(méi)有那么幸運(yùn),這里不糾結(jié),繼續(xù)向下看什么時(shí)候會(huì)觸發(fā)RVO。

示例3: 

  1. std::vector<int> return_vector(void) {  
  2.     std::vector<int> tmp {1,2,3,4,5};  
  3.     return std::move(tmp);  
  4.  
  5. std::vector<int> &&rval_ref = return_vector(); 

和示例1類(lèi)似,std::move一個(gè)臨時(shí)對(duì)象是沒(méi)有必要的,也會(huì)忽略掉返回值優(yōu)化。

最好的代碼: 

  1. std::vector<int> return_vector(void) {  
  2.     std::vector<int> tmp {1,2,3,4,5};  
  3.     return tmp;  
  4.  
  5. std::vector<int> rval_ref = return_vector(); 

這段代碼會(huì)觸發(fā)RVO,不拷貝也不移動(dòng),不生成臨時(shí)對(duì)象。 

 

責(zé)任編輯:龐桂玉 來(lái)源: C語(yǔ)言與C++編程
相關(guān)推薦

2021-02-01 08:39:26

JTAG接口Jlink

2021-07-19 08:33:56

時(shí)間復(fù)雜度大O

2020-12-08 11:08:55

時(shí)間復(fù)雜度軟件

2021-12-09 08:16:40

JVM參數(shù)系統(tǒng)

2022-10-12 08:22:44

Guava工具Collection

2021-07-01 09:00:00

安全數(shù)字化轉(zhuǎn)型滲透

2020-06-02 07:00:00

會(huì)話(huà)安全黑客攻擊

2017-08-29 11:21:03

微軟

2019-11-04 09:07:48

DevOps互聯(lián)網(wǎng)IT

2023-09-11 08:51:23

LinkedList雙向鏈表線(xiàn)程

2021-06-17 13:40:47

區(qū)塊鏈比特幣公有鏈

2021-05-17 07:04:07

動(dòng)態(tài)代理面試

2020-03-18 18:20:19

區(qū)塊鏈數(shù)字貨幣比特幣

2019-04-24 08:31:43

分布式限流kafka

2019-12-25 14:00:26

數(shù)據(jù)科學(xué)人工智能科學(xué)家

2021-12-27 08:00:00

Kubernetes容器安全

2022-09-22 08:00:00

API開(kāi)發(fā)數(shù)據(jù)

2018-11-28 10:39:01

5G網(wǎng)絡(luò)運(yùn)營(yíng)商

2020-12-23 09:00:00

開(kāi)發(fā)Web工具

2011-08-12 09:27:33

移動(dòng)設(shè)備虛擬化虛擬化
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)