2.高級聲明
int?*func(); int?(*func)(); int?*arr[]; int?(*func[])();
? ? 第1行聲明一個返回值為int型指針的函數。()優(yōu)先級高于間接訪問操作符*。
????第2行的第2對括號是函數調用操作符,但第1對括號只起到聚類的作用。它迫使間接訪問在函數調用之前進行,使func成為一個函數指針,它所指向的函數返回一個int值。
? ? 第3行聲明一個數組,元素類型是指向整型的指針。
? ? 第4行func是一個數組,數組元素的類型是函數指針,其指向的函數返回值是一個int值。
3.函數指針
? ? 函數指針常見的用途有轉換表(jump table)和作為參數傳遞給另一個函數。對函數指針執(zhí)行間接訪問之前必須把它初始化為指向某個函數。
int?func(?int?); int?(*pf)(?int?)?=?&func; int?ans; ans?=?func(25); ans?=?(*pf)(?25?); ans?=?pf(?25?);
? ? 調用函數時的執(zhí)行過程如:首先函數名func被轉換成一個函數指針,該指針指定函數在內存中的位置。然后函數調用操作符調用該函數,執(zhí)行開始于這個地址的代碼。所以三個示例效果一樣。
3.1回調函數
int?(*compare_ints)(?void?const?*a,?void?const?*b) { if(?*(int?*)a?==?*(int?*)b) return?0; else return?1; } Node?*search_list(?Node?*node,?void?const?*value,?int?(*compare)(?void?const?*,?void?const?*)) { while(?node?!=?NULL) { if(?compare(?&node->value,?value?)?==?0) break; node?=?node->link; } return?node; } desired_node?=?search_list(?root,?&desired_value,?compare_ints);
? ? 函數search_list的第3個參數是一個函數指針。這個參數用一個完整的原型進行聲明。node若被聲明為const,函數將不得不返回一個const結果,這將限制調用函數,它便無法查找函數所找到的節(jié)點。
desired_node?=?search_list(?root,?&desired_value,?strcmp?);
? ? 若鏈表是字符串鏈表,則上述代碼可以完成比較。
3.2轉移表
? ? 程序其他部分讀入兩個數(op1和op2)和一個操作符。
switch(?oper?) { case?ADD: result?=?add(?op1,?op2?); break; case?SUB: result?=?sub(?op1,?op2?); break; case?MUL: result?=?mul(?op1,?op2?); break case?DIV: result?=?div(?op1,?op2?); break; ... }
? ? 采用調用函數來執(zhí)行這些操作可以體現(xiàn)一種良好的設計方案,即把具體操作和選擇操作的代碼分開。
double?add(?double,?double?); double?sub(?double,?double?); double?mul(?double,?double?); double?div(?double,?double?); ... double?(*oper_func[])(?double,?double?)?=?{ add,?sub,?mul,?div,?... }
4.命令行函數
int?main(?int?argc,?char?**argv?);
5.字符串常量
? ? 當一個字符串常量出現(xiàn)在表達式中時,它的值是個指針常量(常量么?)。編譯器把這些指定字符的一份拷貝存儲在內存的某個位置,并存儲一個指向第1個字符的指針。所以,
"xyz"?+?1
? ? 上面這行代碼的意義是計算“指針值加上1”。結果是一個指針,指向字符串中第2個字符:y。
*"xyz"
? ? 上面這個間接訪問的結果就是它指向的字符:x。
"xyz"[2]
? ? 上面這個表達式也是正確的。因為當數組名用于表達式中時,其值為常量指針。
remainder?=?value?%?16; if(?remainder?<?10) ????putchar(?remainder?+?'0'?); else ????putchar(?remainder?-?10?+?'A');
? ? 上面代碼與下面代碼實現(xiàn)相同的功能。
putchar(?"0123456789ABCDEF"[remainder?%?16]?);
void?binary_to_ascii(?unsigned?int?value?) { ????unsigned?int?quotient; ????quotient?=?value?/?10; ????if(?quotient?!=?0) ????????binary_to_ascii(quotient); ????putchar(?value?%?10?+?'0'); }