1.前置運算子和後置運算子,左值和右值。
其實很久以來一直都沒有怎麼搞清楚左值和右值的區別,只知道左值可以放在等號的左邊,也可以放在等號的右邊,但是右值卻只能放在等號的右邊,然後形成乙個大概直觀的印象,知道怎麼樣做才不出錯而已。不過今天看看c++,卻發現有了點新的體會。
對於表示式a--=5;這樣乙個表示式,明顯是錯誤的,究其原因,是因為執行等號左邊的自減表示式之後,顯示取得a的值,然後才是進行自減操作,所以最終的結果是乙個右值,而且就是a的值(變化前的),於是對於賦值語句 b = a--,自然不會弄錯了。
而對於--a=5;這樣乙個表示式,則明顯就不同了。首先執行的是自減操作,然後返回的是a的值(變化後的),於是自然就可以將返回的a值再次賦值了。也就是說前置的運算子是左值表示式,而後置的則是右值表示式。於是,我想到了很久前的乙個想法:++a++ = 5;當然,行家肯定一看就知道是錯誤的,可是錯誤的地方需要改正啊。於是我想到了幾種方法,就是通過加括號的方法來實現。
1)++(a++) = 5; 其實通過錯誤資訊就可以看到原因了:error: non-lvalue in increment ,明顯說明在自增運算子中的那個表示式不是左值,所以不能自增運算了。哦,對啊,a++是右值,於是的話再次進行++的操作就是錯誤的,因為++操作需要的表示式是左值表示式。於是此處的方法行不通。
2)(++a)++ = 5;此處也不行,錯誤資訊就是: error: non-lvalue in assignment,通過前面的表述就知道了,其實++a是左值表示式,那麼後面的那個++操作就是乙個錯誤了,賦值的時候,等號的左邊不是左值表示式,因為括號外的是乙個後置的++運算子。於是,就得到了錯誤。
通過上面的這些折騰,對於++++a等等**的表示式,相信也就有更深刻的體會了。
2.過載前置++和後置++的時候的不同
相信很多人都做過這種操作了,對於後置的++,需要多乙個形參int來說明,但是從上面的分析,還需要注意的是兩者的左值和右值的區別,過載的目的,也不過是為了更加符合使用者的習慣,於是,我們對於前置的++,自然要返回的是左值表示式,而對於後置的++,需要返回的則是左值表示式。當然,具體的來說說吧。
前置++:如果自定義了乙個類a(含有乙個資料成員x),那麼要過載它的前置++運算子,根據習慣,首先就是對它的資料成員++,然後再返回它的引用,這樣,才是左值表示式,於是下面的**: a& a::operator++()這樣的方法,自然很好的表達了前置的作用,先運算,在返回值。
後置++:如果同上面的例子,要定義後置++的話,那麼需要注意的是它的右值性,此時是返回引用還是類型別呢?思考了很久,一般來說,返回的是引用的,都是左值表示式,而如果此處返回的是引用,那麼就可能在後面的**中出現問題。
於是出現了下面的這樣的**:
a a::operator++(int )
其實我覺得此句甚好,既表達了後置的意思,又成功的返回了值。後置的含義,也就是先返回之後再進行自增操作,於是,此處呼叫的是預設的拷貝建構函式來實現返回乙個臨時的變數。
3.過載函式的引數問題
以前的時候被這個問題糾結的不行了,後來明白了一點,今天看看書,明白的更多了,現在才明白古人那句「書讀百遍,其義自見」的至理名言了。也更加的明白了讀書的重要性。
對於運算子過載,我們常用的,既可以作為成員函式,也可以作為普通函式,這點都知道,而且,此時的引數不同,作為成員函式的話,因為在類中,所以有乙個隱藏的this指標,於是,雙目運算子就只要寫出來乙個就可以,而單目運算子就不需要寫出來。同樣的是乙個過載運算子,可以有兩種表示的方法,如:a operator+(a&, int value); //此為普通的函式 a a::operator+(int value) ; //此為成員函式
對於上面的這兩個典型的例子,很容易而直白的理解,在呼叫的時候,寫下 a + 5; 其中a是a型別的變數,這樣就ok了。但是深入思考下,過載的是+運算子,那麼,雙目的運算,那兩個引數的順序是,如何區分呢?很簡單,前面的那個操作符就是第乙個操作符,而後面的那個就是第二個運算子。對於是成員的函式,那麼就要注意this指標的作用,所以也就說明指代的物件的重要性了。為何我這麼簡單的東西還要思考呢?因為被我們書上的乙個**程式搞懵了,如下:
#include
using namespace std;
struct a};
class b };
a *b::operator->()
int main()
各位可以看看,其實道理很簡單,不過是過載了->,而且過載的比較**,讓人有點琢磨不透而已。對於箭頭運算子,一般的左邊為指標,不過這裡過載的話左邊為乙個類物件,注意看宣告時在b中的,所以左邊的那個就是b型別的變數,也就是b,而這個是單目運算子,於是過載之後返回的指標,照理說應該是b-> –>a,因為左邊做了操作之後返回的是指標,然後再訪問a,不過應該是這裡的編譯器處理或者規定,所以就只剩下了乙個->,從此處,我看出來了過載的引數順序,以及引數的個數。以前總是手動的加上個引數,或者少乙個,現在將this算入其中,然後看是否為成員,就可以得到確定的數目,然後就可以知道具體的呼叫順序了。
ok,先寫到這裡了,還有n多報告沒有寫,先打住吧!
漫談C 過載運算子
1.前置運算子和後置運算子,左值和右值 其實很久以來一直都沒有怎麼搞清楚左值和右值的區別,只知道左值可以放在等號的左邊,也可以放在等號的右邊,但是右值卻只能放在等號的右邊,然後形成乙個大概直觀的印象,知道怎麼樣做才不出錯而已。不過今天看看c 卻發現有了點新的體會。對於表示式a 5 這樣乙個表示式,明...
C 過載運算子
運算子過載是一種形式的c 多型。在c 中,編譯器有能力把乙個由資料 物件和操作符共同組成的表示式,解釋為對乙個全域性或成員函式的呼叫。該全域性或成員函式被稱為操作符函式,通過重定義操作符函式,可以實現針對自定義型別 結構,類 的運算法則,並使之與內建型別一樣參與各種表示式。過載運算子可使 看起來更加...
C 過載運算子
本文主要講述加號運算子 自增運算子 流提取運算子運 流插入運算子 先給出vector類 class vector 建構函式 vector const vector v 拷貝建構函式 vector operator const vector v 過載 vector operator 過載前置自增運算子...