C和指針_第13章_高級(jí)指針話題_學(xué)習(xí)筆記
2.高級(jí)聲明
int?*func(); int?(*func)(); int?*arr[]; int?(*func[])();
? ? 第1行聲明一個(gè)返回值為int型指針的函數(shù)。()優(yōu)先級(jí)高于間接訪問操作符*。
????第2行的第2對(duì)括號(hào)是函數(shù)調(diào)用操作符,但第1對(duì)括號(hào)只起到聚類的作用。它迫使間接訪問在函數(shù)調(diào)用之前進(jìn)行,使func成為一個(gè)函數(shù)指針,它所指向的函數(shù)返回一個(gè)int值。
? ? 第3行聲明一個(gè)數(shù)組,元素類型是指向整型的指針。
? ? 第4行func是一個(gè)數(shù)組,數(shù)組元素的類型是函數(shù)指針,其指向的函數(shù)返回值是一個(gè)int值。
3.函數(shù)指針
? ? 函數(shù)指針常見的用途有轉(zhuǎn)換表(jump table)和作為參數(shù)傳遞給另一個(gè)函數(shù)。對(duì)函數(shù)指針執(zhí)行間接訪問之前必須把它初始化為指向某個(gè)函數(shù)。
int?func(?int?); int?(*pf)(?int?)?=?&func; int?ans; ans?=?func(25); ans?=?(*pf)(?25?); ans?=?pf(?25?);
? ? 調(diào)用函數(shù)時(shí)的執(zhí)行過程如:首先函數(shù)名func被轉(zhuǎn)換成一個(gè)函數(shù)指針,該指針指定函數(shù)在內(nèi)存中的位置。然后函數(shù)調(diào)用操作符調(diào)用該函數(shù),執(zhí)行開始于這個(gè)地址的代碼。所以三個(gè)示例效果一樣。
3.1回調(diào)函數(shù)
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);
? ? 函數(shù)search_list的第3個(gè)參數(shù)是一個(gè)函數(shù)指針。這個(gè)參數(shù)用一個(gè)完整的原型進(jìn)行聲明。node若被聲明為const,函數(shù)將不得不返回一個(gè)const結(jié)果,這將限制調(diào)用函數(shù),它便無法查找函數(shù)所找到的節(jié)點(diǎn)。
desired_node?=?search_list(?root,?&desired_value,?strcmp?);
? ? 若鏈表是字符串鏈表,則上述代碼可以完成比較。
3.2轉(zhuǎn)移表
? ? 程序其他部分讀入兩個(gè)數(shù)(op1和op2)和一個(gè)操作符。
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; ... }
? ? 采用調(diào)用函數(shù)來執(zhí)行這些操作可以體現(xiàn)一種良好的設(shè)計(jì)方案,即把具體操作和選擇操作的代碼分開。
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.命令行函數(shù)
int?main(?int?argc,?char?**argv?);
5.字符串常量
? ? 當(dāng)一個(gè)字符串常量出現(xiàn)在表達(dá)式中時(shí),它的值是個(gè)指針常量(常量么?)。編譯器把這些指定字符的一份拷貝存儲(chǔ)在內(nèi)存的某個(gè)位置,并存儲(chǔ)一個(gè)指向第1個(gè)字符的指針。所以,
"xyz"?+?1
? ? 上面這行代碼的意義是計(jì)算“指針值加上1”。結(jié)果是一個(gè)指針,指向字符串中第2個(gè)字符:y。
*"xyz"
? ? 上面這個(gè)間接訪問的結(jié)果就是它指向的字符:x。
"xyz"[2]
? ? 上面這個(gè)表達(dá)式也是正確的。因?yàn)楫?dāng)數(shù)組名用于表達(dá)式中時(shí),其值為常量指針。
remainder?=?value?%?16; if(?remainder?<?10) ????putchar(?remainder?+?'0'?); else ????putchar(?remainder?-?10?+?'A');
? ? 上面代碼與下面代碼實(shí)現(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'); }