逗號操作符重載分析(三十三)
下來我們以代碼為例進行逗號表達式的說明
#include <iostream> #include <string> using namespace std; void func(int i) { cout << "void func(int i): " << i << endl; } int main() { int i = 0; int j = 0; int a[3][3] = { (0, 1, 2), (3, 4, 5), (6, 7, 8) }; while( i < 5 ) func(i), i++; for(i=0; i<3; i++) { for(j=0; j<3; j++) { cout << a[i][j] << endl; } } (i, j) = 5; cout << "i = " << i << endl; cout << "j = " << j << endl; return 0; }
那麽我們在第 21 行定義了一個 while 循環,它的作用域只有下面的一行,按理說它會是個死循環。接下來我們想要打印數組 a 的值,應該打印的是我們定義的值,那麽第 34 行這樣的賦值會成功嗎?我們來看看編譯結果
我們看到並沒有執行死循環,而是執行了 5 次就退出了。接下來的數組打印也只是打印了 2,5,8。別的都沒有打印出來,最後一個是執行成功的。那麽我們來分析下,仔細看看第 22 行後面的是逗號,並不是我們平時所定義的分號。所以它和下一句 i++ 結合在一起了,因此便不會是死循環。我們定義數組的時候是 (),而不是 {},因此它將數組中的值當成逗號表達式處理了。最後一個賦值操作也會是當逗號表達式處理,應該是 j 被賦值為 5,經過前面 for 循環,i 則為 3 了。我們接下來將數組 a 中的 () 改成 {} 試試
我們看到它已經成功執行了。那麽在 C++ 中重載逗號操作符時合法的,使用全局函數對逗號操作符進行重載。重載函數的參數必須有一個是類類型,重載函數的返回值類型必須是引用。
下來我們以代碼為例進行分析
#include <iostream> #include <string> using namespace std; class Test { int mValue; public: Test(int i) { mValue = i; } int value() { return mValue; } }; Test& operator , (const Test& a, const Test& b) { return const_cast<Test&>(b); return i; } Test func(Test& i) { cout << "func(): i =" << i.value() << endl; } int main() { Test t0(0); Test t1(1); Test tt = (t0, t1); cout << tt.value() << endl; return 0; }
我們來看看編譯結果
這樣看來也沒錯嘛,那麽我們在第 36 行用函數的形式來調用下,別忘了重載的本質就是函數調用。
雖然結果也是對的,但是我們有沒有發現它不是從左向右依次執行的,這就和逗號表達式的本意相反了。在 C++ 中通過函數調用擴展操作符的功能後,進入函數體前必須完成所有參數的計算。而函數參數的計算次序是不定的,這便導致了重載後無法嚴格從左向右計算表達式。所以在工程中,一般是不要重載逗號操作符!!!
通過對逗號操作符重載的學習,總結如下:1、逗號表達式從左向右順序計算每個子表達式的值;2、逗號表達式的值為最後一個子表達式的值;3、操作符重載無法完全實現逗號操作符的原生意義;4、工程開發中不要重載逗號操作符。
歡迎大家一起來學習 C++ 語言,可以加我QQ:243343083。
逗號操作符重載分析(三十三)