c++中臨時變量不能作為非const的引用參數(shù)的方法
試看下面的代碼:
#include <iostream> using namespace std; void f(int &a) { cout << "f(" << a << ") is being called" << endl; } void g(const int &a) { cout << "g(" << a << ") is being called" << endl; } int main() { int a = 3, b = 4; f(a + b); //編譯錯誤,把臨時變量作為非const的引用參數(shù)傳遞了 g(a + b); //OK,把臨時變量作為const&傳遞是允許的 }
上面的兩個調(diào)用之前,a+b的值會存在一個臨時變量中,當(dāng)把這個臨時變量傳給f時,由于f的聲明中,參數(shù)是int&,不是常量引用,所以產(chǎn)生以下編譯錯誤:
const_ref.cpp: In function `int main()': const_ref.cpp:14: error: invalid initialization of non-const reference of type ' int&' from a temporary of type 'int' const_ref.cpp:4: error: in passing argument 1 of `void f(int&)' 而在g(a+b)中,由于g定義的參數(shù)是const int&,編譯通過。 問題是為什么臨時變量作為引用參數(shù)傳遞時,必須是常量引用呢?很多人對此的解釋是臨時變量是常量,不允許賦值,改動,所以當(dāng)作為非常量引用傳遞時,編譯器就會報錯。這個解釋在關(guān)于理解臨時變量不能作為非const引用參數(shù)這個問題上是可以的,但不夠準(zhǔn)確。事實(shí)上,臨時變量是可以被作為左值(LValue)并被賦值的,請看下面的代碼: #include <iostream> using namespace std; class CComplex { friend CComplex operator+(const CComplex &cp1, const CComplex &cp2); friend ostream& operator<<(ostream &os, const CComplex &cp); private: int x; public: CComplex(){} CComplex(int x1) { x = x1; } }; CComplex operator+(const CComplex &cp1, const CComplex &cp2) { CComplex cp3; cp3.x = cp1.x + cp2.x; return cp3; } ostream& operator<<(ostream &os, const CComplex &cp) { os << cp.x; return os; } int main() { CComplex a(2), b(3), c(4); cout << (a + b) << endl; cout << ((a + b) = c) << endl; //臨時對象作為左值 return 0; }
上面的程序編譯通過,而且運(yùn)行結(jié)果是:
5
4
臨時變量確實(shí)被賦值,而且成功了。
所以,臨時變量不能作為非const引用參數(shù),不是因?yàn)樗浅A?,而是因?yàn)閏++編譯器的一個關(guān)于語義的限制。如果一個參數(shù)是以非const引用傳入,c++編譯器就有理由認(rèn)為程序員會在函數(shù)中修改這個值,并且這個被修改的引用在函數(shù)返回后要發(fā)揮作用。但如果你把一個臨時變量當(dāng)作非const引用參數(shù)傳進(jìn)來,由于臨時變量的特殊性,程序員并不能操作臨時變量,而且臨時變量隨時可能被釋放掉,所以,一般說來,修改一個臨時變量是毫無意義的,據(jù)此,c++編譯器加入了臨時變量不能作為非const引用的這個語義限制,意在限制這個非常規(guī)用法的潛在錯誤。
還不明白?OK,我們說直白一點(diǎn),如果你把臨時變量作為非const引用參數(shù)傳遞,一方面,在函數(shù)申明中,使用非常量型的引用告訴編譯器你需要得到函數(shù)對某個對象的修改結(jié)果,可是你自己又不給變量起名字,直接丟棄了函數(shù)的修改結(jié)果,編譯器只能說:“大哥,你這是干啥呢,告訴我把結(jié)果給你,等我把結(jié)果給你了,你又直接給扔了,你這不是在玩我呢嗎?”所以編譯器一怒之下就不讓過了。這下大家明白了吧?
以上這篇c++中臨時變量不能作為非const的引用參數(shù)的方法就是小編分享給大家的全部內(nèi)容了,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
C++基于單鏈表實(shí)現(xiàn)學(xué)生成績管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C++基于單鏈表實(shí)現(xiàn)學(xué)生成績管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-05-05C語言實(shí)現(xiàn)個人通訊錄管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C語言實(shí)現(xiàn)個人通訊錄管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2019-12-12Clion(CMake工具)中引入第三方庫的詳細(xì)方法
這篇文章主要介紹了Clion(CMake工具)中引入第三方庫的詳細(xì)方法,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-02-02opencv實(shí)現(xiàn)圖像顏色空間轉(zhuǎn)換
這篇文章主要為大家詳細(xì)介紹了opencv實(shí)現(xiàn)圖像顏色空間轉(zhuǎn)換,具有一定的參考價值,感興趣的小伙伴們可以參考一下2019-08-08C++超詳細(xì)實(shí)現(xiàn)堆和堆排序過像
堆是計算機(jī)科學(xué)中一類特殊的數(shù)據(jù)結(jié)構(gòu)的統(tǒng)稱,通常是一個可以被看做一棵完全二叉樹的數(shù)組對象。而堆排序是利用堆這種數(shù)據(jù)結(jié)構(gòu)所設(shè)計的一種排序算法。本文將通過圖片詳細(xì)介紹堆排序,需要的可以參考一下2022-06-06VS2019編寫C程序或者CUDA程序出現(xiàn)“無法啟動程序,系統(tǒng)找不到指定的文件”問題的詳細(xì)解決方法
這篇文章主要介紹了VS2019編寫C程序或者CUDA程序出現(xiàn)“無法啟動程序,系統(tǒng)找不到指定的文件”問題的詳細(xì)解決方法,文中通過圖文的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08C/C++高精度運(yùn)算(大整數(shù)運(yùn)算)詳細(xì)講解
高精度算法的本質(zhì)是把大數(shù)拆成若干固定長度的塊,然后對每一塊進(jìn)行相應(yīng)的運(yùn)算,下面這篇文章主要給大家介紹了關(guān)于C/C++高精度運(yùn)算(大整數(shù)運(yùn)算)的相關(guān)資料,需要的朋友可以參考下2022-11-11C++在多線程中使用condition_variable實(shí)現(xiàn)wait
這篇文章主要介紹了C++中的condition_variable中在多線程中的使用,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)吧2022-09-09