C陷阱与缺陷代码分析之第2章语法陷阱
作者:劉昊昱?
博客:http://blog.csdn.net/liuhaoyutz
?
陷阱1 理解函數聲明
作者提出一個問題:有一個首地址為0的函數,該函數返回值類型為void,沒有參數。怎樣用C語言的語句調用這個函數?
答案是(*(void? (*)())0)();
?
要理解這個調用形式,要清楚如下兩個問題:
一是函數指針。
假設fp是一個函數指針,則調用fp所指向的函數的方法是
(*fp)();
因為fp是一個函數指針,所以*fp是該指針所指向的函數,所以(*fp)()就是調用該函數的方式。ANSI C允許將(*fp)()簡寫為fp(),fp()也是我們比較常見的形式,但是一定要知道這種寫法是一種簡寫形式。例如prinf()函數,printf就是函數指針,它的完整形式是(*printf)()。為了說明這個問題,我們來看一個測試程序page17.c,代碼如下:
?
1#include <stdio.h> 2 3int main() 4{ 5 printf("test1\n"); 6 (*printf)("test2\n"); 7 8 return 0; 9}?
?
編譯運行結果如下:
?
二是強制類型轉換符的聲明方式。
某類型的強制類型轉換符,只需要把該類型變量聲明中的變量名和聲明末尾的分號去掉,再將剩余的部分用一個括號封裝起來即可。例如,聲明一個int型指針變量的方式是
int *p;
按照上面的原則,int型指針強制類型轉換符就是把變量名p和末尾的分號去掉,再把剩余的部分用一個括號封裝起來,即(int *)。
同理,聲明一個返回值為void,沒有參數的函數指針變量f的方式是:
void (*f)();
按照上面的原則,返回值為void,沒有參數的函數指針類型強制轉換符就是把變量名f和最后的分號去掉,再把剩余的部分用一個括號封裝起來,即(void (*)())。
?
有了上面的預備知識,我們可以來看作者提出的問題了。首地址為0的函數,也就是函數指針的值為0,函數返回值類型為void,沒有參數。所以我們把0強制轉換為(void (*)())類型就是該函數的函數指針,有了函數指針,要調用該函數,則是(*(void (*)())0)();
如果使用typedef能夠使表述更加清晰:
typedef void (*funcptr)();
(*(funcptr)0)();
?
作者舉的第二個例子是signal函數,其函數聲明如下:
void (*signal(int, void(*)(int)))(int);
怎樣來理解這個函數聲明呢?
signal函數有兩個參數,第一個參數是一個整數,代表需要“被捕獲”的特定信號。第二個參數是一個函數指針,它是信號處理函數指針,它的返回值類型為void,該信號處理函數同樣有一個int型參數代表要處理的信號。
讓我們從信號處理函數開始,信號處理函數的函數指針聲明如下:
void (*sfp)(int);
信號處理函數指針類型可以通過把指針變量名sfp和最后的分號去掉得到,即:
void (*)(int)
signal函數的返回值是原來的信號處理函數指針,即singnal函數的返回值類型是void(*)(int)。
綜上所述可知,signal函數的聲明形式應該是:
void (*signal(int, void(*)(int)))(int);
同樣地,使用typedef可以簡化signal函數的聲明:
typedef void (*HANDLER)(int);
HANDLER signal (int, HANDLER);
?
陷阱二運算符的優先級
記住C語言運算符的優先級是非常有益的,但是,C語言運算符優先級多達15個,記住它們并不是一件容易的事。完整的C語言運算符優先級如下表所示:
| 優先級 | 運算符 | 名稱或含義 | 使用形式 | 結合方向 | 說明 |
| 1 | [] | 數組下標 | 數組名[常量表達式] | 左到右 | |
| () | 圓括號 | (表達式)/函數名(形參表) | |||
| . | 成員選擇(對象) | 對象.成員名 | |||
| -> | 成員選擇(指針) | 對象指針->成員名 | |||
| 2 | - | 負號運算符 | -表達式 | 右到左 | 單目運算符 |
| (類型) | 強制類型轉換 | (數據類型)表達式 | |||
| ++ | 自增運算符 | ++變量名/變量名++ | 單目運算符 | ||
| -- | 自減運算符 | --變量名/變量名-- | 單目運算符 | ||
| * | 取值運算符 | *指針變量 | 單目運算符 | ||
| & | 取地址運算符 | &變量名 | 單目運算符 | ||
| ! | 邏輯非運算符 | !表達式 | 單目運算符 | ||
| ~ | 按位取反運算符 | ~表達式 | 單目運算符 | ||
| sizeof | 長度運算符 | sizeof(表達式) | |||
| 3 | / | 除 | 表達式/表達式 | 左到右 | 雙目運算符 |
| * | 乘 | 表達式*表達式 | 雙目運算符 | ||
| % | 余數(取模) | 整型表達式/整型表達式 | 雙目運算符 | ||
| 4 | + | 加 | 表達式+表達式 | 左到右 | 雙目運算符 |
| - | 減 | 表達式-表達式 | 雙目運算符 | ||
| 5 | <<? | 左移 | 變量<<表達式 | 左到右 | 雙目運算符 |
| >>? | 右移 | 變量>>表達式 | 雙目運算符 | ||
| 6 | >? | 大于 | 表達式>表達式 | 左到右 | 雙目運算符 |
| >= | 大于等于 | 表達式>=表達式 | 雙目運算符 | ||
| <? | 小于 | 表達式<表達式 | 雙目運算符 | ||
| <= | 小于等于 | 表達式<=表達式 | 雙目運算符 | ||
| 7 | == | 等于 | 表達式==表達式 | 左到右 | 雙目運算符 |
| != | 不等于 | 表達式!= 表達式 | 雙目運算符 | ||
| 8 | & | 按位與 | 表達式&表達式 | 左到右 | 雙目運算符 |
| 9 | ^ | 按位異或 | 表達式^表達式 | 左到右 | 雙目運算符 |
| 10 | | | 按位或 | 表達式|表達式 | 左到右 | 雙目運算符 |
| 11 | && | 邏輯與 | 表達式&&表達式 | 左到右 | 雙目運算符 |
| 12 | || | 邏輯或 | 表達式||表達式 | 左到右 | 雙目運算符 |
| 13 | ?: | 條件運算符 | 表達式1? 表達式2: 表達式3 | 右到左 | 三目運算符 |
| 14 | = | 賦值運算符 | 變量=表達式 | 右到左 | |
| /= | 除后賦值 | 變量/=表達式 | |||
| *= | 乘后賦值 | 變量*=表達式 | |||
| %= | 取模后賦值 | 變量%=表達式 | |||
| += | 加后賦值 | 變量+=表達式 | |||
| -= | 減后賦值 | 變量-=表達式 | |||
| <<= | 左移后賦值 | 變量<<=表達式 | |||
| >>= | 右移后賦值 | 變量>>=表達式 | |||
| &= | 按位與后賦值 | 變量&=表達式 | |||
| ^= | 按位異或后賦值 | 變量^=表達式 | |||
| |= | 按位或后賦值 | 變量|=表達式 | |||
| 15 | , | 逗號運算符 | 表達式,表達式,… | 左到右 | 從左向右順序運算 |
如果把這些運算符恰當分組,并且理解了各組運算符之間的相對優先級,那么這張表其實不難記住。
第一:
優先級最高者其實并不是真正意義上的運算符,包括:數組下標、函數調用操作符,結構成員選擇符。它們的優先級是1級,都是自左向右結合,因此a.b.c的含義是(a.b).c,而不是a.(b.c)。
第二:
單目運算符的優先級僅次于前述運算符,它們的優先級是2級。在所有真正意義上的運算符中,它們的優先級最高。單目運算符是從右向左結合的,因此*p++會被編譯器解釋成*(p++),即取指針p所指向的對象,然后將p遞增1,而不是(*p)++,即取指針p所指向的對象,然后將該對象的值加1。
第三:
優先級比單目運算符低的,接下來就是雙目運算符。
在雙目運算符中,算術運算符優先級最高(乘、除、取余為3級,加、減為4級),
移位運算符次之(左移>>、右移<<,為5級),
關系運算符再次之(如>、<、<=等等,為6級,==和!=,為7級),
接著是邏輯運算符(如按位與&、按位或|、邏輯與&&、邏輯或||,等等),
接下來是條件運算符(?:其實這是一個三目運算符),
賦值運算符(=,/=,*=等等),
優先級最低的是逗號運算符(,)。
?
我們需要記住的最重要的一點是:
算術運算符(加減乘除)>
移位運算符(左移>>、右移<<) >
關系運算符(大于>、小于<、等于==,等等) >
邏輯運算符(按位與&、按位或|、邏輯與&&、邏輯或||,等等)
?
?
轉載于:https://www.cnblogs.com/dyllove98/p/3199076.html
總結
以上是生活随笔為你收集整理的C陷阱与缺陷代码分析之第2章语法陷阱的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 怎么用命令开远程主机的telnet服务
- 下一篇: 枚举举例