上表中可以總結出如下規律:
- 結合方向只有三個是從右往左,其余都是從左往右。
- 所有雙目運算符中只有賦值運算符的結合方向是從右往左。
- 另外兩個從右往左結合的運算符也很好記,因為它們很特殊:一個是單目運算符,一個是三目運算符。
- C語言中有且只有一個三目運算符。
- 逗號運算符的優先級最低,要記住。
- 此外要記住,對於優先級:算術運算符 > 關系運算符 > 邏輯運算符 > 賦值運算符。邏輯運算符中“邏輯非 !”除外。
一些容易出錯的優先級問題
上表中,優先級同為1 的幾種運算符如果同時出現,那怎么確定表達式的優先級呢?這是很多初學者迷糊的地方。下表就整理了這些容易出錯的情況:
優先級問題 | 表達式 | 經常誤認為的結果 | 實際結果 |
---|---|---|---|
. 的優先級高於 *(-> 操作符用於消除這個問題) | *p.f | p 所指對象的字段 f,等價於: (*p).f |
對 p 取 f 偏移,作為指針,然后進行解除引用操作,等價於: *(p.f) |
[] 高於 * | int *ap[] | ap 是個指向 int 數組的指針,等價於: int (*ap)[] |
ap 是個元素為 int 指針的數組,等價於: int *(ap []) |
函數 () 高於 * | int *fp() | fp 是個函數指針,所指函數返回 int,等價於: int (*fp)() |
fp 是個函數,返回 int*,等價於: int* ( fp() ) |
== 和 != 高於位操作 | (val & mask != 0) | (val &mask) != 0 | val & (mask != 0) |
== 和 != 高於賦值符 | c = getchar() != EOF | (c = getchar()) != EOF | c = (getchar() != EOF) |
算術運算符高於位移 運算符 | msb << 4 + lsb | (msb << 4) + lsb | msb << (4 + lsb) |
逗號運算符在所有運 算符中優先級最低 | i = 1, 2 | i = (1,2) | (i = 1), 2 |
這些容易出錯的情況,希望讀者好好在編譯器上調試調試,這樣印象會深一些。一定要多調試,光靠看代碼,水平是很難提上來的。調試代碼才是最長水平的。
轉 : http://c.biancheng.net/view/161.html