分類和級別
C語言的運算符主要用於構成表達式,同一個符號在不同的表達式中,其作用並不一致。下面按計算的優先順序,分別說明不同作用的表達式。需要特別指出,在C語言標準中,並沒有結合性的說法。
相同優先權運算符,從左至右依次運算。注意後綴運算優先權高於前綴。因此++i++應解釋為++(i++)。
而與或非的運算優先權都不一樣,因此a && b || b && c解釋為(a && b) || (b && c)
合理使用優先權可以極大簡化表達式。
基本表達式 1級
基本表達式(Primary expressions),主要是用於運算符之間,做為運算數。
標識,常量,字元串文字量,優先權提升表達式最優先執行。
優先權提升表達式是指圓括弧包圍的表達式,如“( expression )”
後綴表達式 2級
postfix-expression [ expression ],數組下標運算。
postfix-expression ( argument-expression-list),函式調用,括弧內的參數可選。
postfix-expression . identifier,成員訪問,
postfix-expression -> identifier,成員訪問,->號之前應為指針。
postfix-expression ++,後綴自增
postfix-expression --,後綴自減
( type-name ) { initializer-list }
( type-name ) { initializer-list , } 複合初始化,C99後新增。例如
單目/一元運算 3級
++ unary-expression 前綴自增
-- unary-expression 前綴自減
unary-operator cast-expression 單目轉型表達式, 包括 取地址& ,提領 * , 正號+ ,負號- 位反~ 邏輯否!。
sizeof unary-expression 求類型長度,對表達式求類型長度
sizeof ( type-name ) 求類型長度
強制類型表達式 4級
( type-name ) cast-expression,強制表達式成為type-name指定的類型。
乘法表達式 5級
“ * ” 乘法運算符;“ / ”除法運算符;“ % ” 取余運算符。
加法運算符 6級
“ + ”加法運算符;“ - ”減法運算符。
移位運算符 7級
<< 左移運算符;>> 右移運算符。
關係運算符 8級
<、<=、>、>=關係運算符。
相等運算符 9級
“ == ”等於運算符;“ != ”不等於運算符。
位與運算符 10級
“ & ”按位與運算符
位異或運算符 11級
“ ∧ ”按位異或運算符(Bitwise exclusive OR operator)。
位或運算符 12 級
“ | ”按位或運算符(Bitwise inclusive OR operator)。
邏輯與運算符 13級
“&&”邏輯與運算符。
邏輯或運算符 14 級
“ || ”邏輯或運算符。
三元條件運算符 15級
? :條件運算符。
賦值運算符 16 級
=、 +=、 -=、 *=、 /=、 %=、 &=、 ^=、 |=、 <<=、 >>=賦值運算符。
逗號運算符 17級
“,”逗號運算符。
[pre]C 語言中,逗號(,)也可以是運算符,稱為逗號運算符(Comma Operator)。逗號運算符可以把兩個以上(包含兩個)的表達式連線成一個表達式,稱為逗號表達式。其一般形式為:
子表達式1, 子表達式2, ..., 子表達式n
例如:
a + b, c = b, c++
逗號運算符的優先權是所有運算符中級別最低的,通常配合 for 循環使用。逗號表達式最右邊的子表達式的值即為逗號表達式的值。上例中,c++ 的值(c 自增之前的值)即為該表達式的值。
逗號運算符保證左邊的子表達式運算結束後才進行右邊的子表達式的運算。也就是說,逗號運算符是一個序列點,其左邊所有副作用都結束後,才對其右邊的子表達式進行運算。因此,上例中,c 得到 b 的值後,才進行自增運算。
優先權
優先權:C語言中,運算符的運算優先權共分為15 級。1 級最高,15 級最低。 在表達式中,優先權較高的先於優先權較低的進行運算。而在一個運算量兩側的運算符 優先權相同時,則按運算符的結合性所規定的結合方向處理。
結合性:C語言中各運算符的結合性分為兩種,即左結合性(自左至右)和右結合性(自右至左)。例如算術運算符的結合性是自左至右,即先左後右。如有表達式x-y+z 則y 應先與“-”號結合,執行x-y 運算,然後再執行+z 的運算。這種自左至右的結合 方向就稱為“左結合性”。而自右至左的結合方向稱為“右結合性”。最典型的右結合 性運算符是賦值運算符。如x=y=z,由於“=”的右結合性,應先執行y=z 再執行x=(y=z)運算。C語言運算符中有不少為右結合性,應注意區別,以避免理解錯誤。
優先權從上到下依次遞減,最上面具有最高的優先權,逗號操作符具有最低的優先權。
所有的優先權中,只有三個優先權是從右至左結合的,它們是單目運算符、條件運算符、賦值運算符。其它的都是從左至右結合。
具有最高優先權的其實並不算是真正的運算符,它們算是一類特殊的操作。()是與函式相關,[]與數組相關,而->及.是取結構成員。
其次是單目運算符,所有的單目運算符具有相同的優先權,因此在我認為的 真正的運算符中它們具有最高的優先權,又由於它們都是從右至左結合的,因此*p++與*(p++)等效是毫無疑問的。
另外在C語言裡,沒有前置後置之分,因為++ -- 是右結合所以右側優先運算,表現為 "運算元後置優先權比較高" 的假象,前置和後置的區分是因為運算符重載而後加入C++的
接下來是算術運算符,*、/、%的優先權當然比+、-高了。
移位運算符緊隨其後。
其次的關係運算符中,< <= > >=要比 == !=高一個級別,不大好理解。
所有的邏輯操作符都具有不同的優先權(單目運算符除外,!和~)
邏輯位操作符的"與"比"或"高,而"異或"則在它們之間。
跟在其後的&&比||高。
接下來的是條件運算符,賦值運算符及逗號運算符。
在C語言中,只有4個運算符規定了運算方向,它們是&&、| |、條件運算符及賦值運算符。
&&、| |都是先計算左邊表達式的值,當左邊表達式的值能確定整個表達式的值時,就不再計算右邊表達式的值。如 a = 0 && b; &&運算符的左邊位0,則右邊表達式b就不再判斷。
在條件運算符中。如a?b:c;先判斷a的值,再根據a的值對b或c之中的一個進行求值。
賦值表達式則規定先對右邊的表達式求值,因此使 a = b = c = 6;成為可能。
口訣注釋
優先權等級口訣
圓方括弧、箭頭一句號, 自增自減非反負、針強地址長度,
乘除,加減,再移位,
小等大等、等等不等,
八位與,七位異,六位或,五與,四或,三疑,二賦,一真逗。
其中“,”號為一個等級分段。
優先權等級注釋
“圓方括弧、箭頭一句號”指的是第1級的運算符。其中圓方括弧很明顯“()、[]”,箭頭 指的是指向結構體成員運算符“->”,句號 指的是結構體成員運算符“.” ;
“自增自減非反負、針強地址長度”指的是第2級的運算符。其中 非 指的是邏輯運算符“!”,反 指的是按位取反運算符“~”,負 指的是負號運算符“-”,針 指的是指針運算符“*”,強 指的是強制類型轉換運算符,地址 指的是地址運算符“&”,長度 指的是長度運算符“sizeof ”;
“乘除,加減,再移位”移位指的是左移運算符“<<”和右移運算符“>>”,其中除法還包括了 取余運算符“%”;
“小等大等、等等不等” 指的是第6級到第7級的運算符:<、<=、>和>=,等等指的是等於運算符==,不等指的是不等於運算符!=
“八位與,七位異,六位或”其中 八位與 指的是第8級的 按位與 運算符“&”,七位異 指的是第9級的按位異或運算符“^”,六位或 指的是第10級的按位或運算符“ |”;
“五與,四或”指的是第11級、第12級的邏輯與運算符“&&”和邏輯或運算符“||”;
“三疑,二賦,一真逗”指的是第13級到第15級的運算符。其中,三疑指的是條件運算符“?:” (三有雙重含義:即指優先權別是三,它的運算符類型也是三目,疑也取“?”之意),二賦 指的是賦值運算符=、+=、-=、*=、/=、%=、>>=、<<=、&=、^=和|= ,一真逗 指的是第15級的“,”運算符,真字只是為了語句需要罷了。
由於C語言的運算符優先權與C++的不完全一樣(主要是增加了幾個運算符),所以這個口訣不能完全實用於C++.但是應該能夠兼容,大家可以比較一下他們的區別應該就能夠很快掌握C++的優先權的!
套用舉例
1、賦值運算符:a=5;
a=b=0;
第一個賦值語句把5賦給變數a;第二個賦值語句的意思是把0同時賦值給兩個變數。這是因為賦值語句是從右向左運算的,也就是說從右端開始計算,先b=0,然後a=b。
2、複合賦值運算符:a=1;a+=3;
上面第二個賦值語句等價於a=a+3;即a=4。
3、算術運算符:Area=Height*Width;num=num1+num2/num3-num4;
第一個賦值語句Height和Width相乘結果賦給變數Area;第二個賦值語句先完成num2與num3的整除運算,然後與num1相加,再減去num4,結果賦給num。運算符運算順序先算乘除再算加減。單目正和單目負最先運算。
4、邏輯運算符:a=1,b=1;
a||b-1;
因為a=1為真值,所以不管b-1是不是真值,總的表達式一定為真值,這時後面的表達式就不會再計算了。
5、關係運算符:if(a>0)...
如果a>0,則執行if語句中的內容,否則退出。
6、條件運算符:a=(b>0)?b:-b;
當b>0時,a=b;當b不大於0時,a=-b;其實上面的意思就是把b的絕對值賦值給a。
7、逗號運算符:b=2,c=7,d=5;
a=(++b,c--,d+3);
有三個表達式,用逗號分開,所以最終的值應該是最後一個表達式的值,也就是d+3=8,所以a=8。
8、位邏輯運算符
包括:1。&位與符 2。|位或符 3。^位異或符 4。~位取反符
以運算元12為例。位運算符將數字12視為1100。位運算符將運算元視為位而不是數值。數值
可以是任意進制的:十進制、八進制或十六進制。位運算符則將運算元轉化為二進制,並相應地返回1或0。
位運算符將數字視為二進制值,並按位進行相應運算,運算完成後再重新轉換為數字。例如:
表達式10&15表示(1010 & 1111),它將返回表示1010的值10。因為真真得真,或者是11得1,同位全是1結果也是1
表達式10|15表示(1010 | 1111),它將返回表示1111的值15。假假得假。全零得零。
表達式10^15表示(1010 ^ 1111), 它將返回表示0101的值5。此時是同性相斥,相同的就為假。
表達式~10表示(~1010),它將返回表示0101的值 5。此號好理解,按位取反。