- 引用是一個變量,它引用其他變量的內存位置
int x = 34;
int &lRef = x;
- 標識符 IRef 就是一個引用。在聲明中,引用是通過 & 來指示的,出現在類型與變量的標識符之間,這種類型的引用稱為左值引用
- 左值看作是一個關聯了名稱的內存位置,允許程序的其他部分來訪問它,?"名稱" 解釋為任何可用于訪問內存位置的表達式
- 右值則是一個臨時值,它不能被程序的其他部分訪問
int square(int a)
{return a * a;
}
int main()
{int x = 0; // 1x = 12; // 2cout << x << endl; // 3x = square(5); // 4cout << x << endl; // 5return 0;
}
- x 是一個左值,這是因為 x 代表一個內存位置,它可以被程序的其他部分訪問,例如上面注釋的第 2、3、4 和 5 行
- 表達式 square(5) 卻是一個右值,因為它代表了一個由編譯器創建的臨時內存位置,以保存由函數返回的值。該內存位置僅被訪問一次,也就是在第 4 行賦值語句的右側。在此之后,它就會立即被刪除,再也不能被訪問了。對于包含右值的內存位置來說,其本質就是:它雖然沒有名稱,但是可以從程序的其他部分訪問到它
- C++11 引入了右值引用的概念,以表示一個本應沒有名稱的臨時對象。右值引用的聲明與左值引用類似,但是它使用的是 2 個?
&
?符號(&&)
int && rRef = square(5);
cout << rRef << endl;
- 聲明一個右值引用,給一個臨時內存位置分配一個名稱,這使得程序的其他部分訪問該內存位置成為了可能,并且可以將這個臨時位置變成一個左值。
- 右值引用不能約束到左值上,所以,以下代碼將無法編譯:
int x = 0;
int && rRefX = x;
- 初始化完成之后,這個包含值 square(5) 的內存位置有了一個名稱,即 rRef1,所以 rRef1 本身變成了一個左值。這意味著后面的這個初始化語句將不會編譯:int && rRef2 = rRef1;究其原因,就是右側的 rRef1 不再是一個右值。綜上所述,臨時對象最多可以有一個左值引用指向它。如果函數有一個臨時對象的左值引用,則可以確認,程序的其他部分都不能訪問相同的對象
int && rRef1 = square(5);
- 右值一般是不可尋址的常量,或在表達式求值過程中創建的無名臨時對象,短暫性的。
- 左值是可尋址的變量,有持久性;
- 左值和右值主要的區別之一是左值可以被修改,而右值不能。
- 左值引用:引用一個對象;
- 右值引用:就是必須綁定到右值的引用,C++11中右值引用可以實現“移動語義”,通過 && 獲得右值引用。
-
右值引用和相關的移動語義是C++11標準中引入的最強大的特性之一,通過std::move()可以避免無謂的復制,提高程序性能。
特殊情況
- 但const左值引用除外,由于const的不可變性,所以const引用可以指向右值,我們經常使用const引用作為函數參數傳遞? ? const int &e = a*3; // 正確:左值引用,const引用可以綁定到一個右值上
例子
int x = 6; // x是左值,6是右值
int &y = x; // 左值引用,y引用xint &z1 = x * 6; // 錯誤,x*6是一個右值
const int &z2 = x * 6; // 正確,可以將一個const引用綁定到一個右值int &&z3 = x * 6; // 正確,右值引用
int &&z4 = x; // 錯誤,x是一個左值
參考鏈接
- C++左值和右值(詳解版)
- 一篇文章弄懂C++左值引用和右值引用 - html中文網