函數(shù)在內(nèi)存中有一個(gè)物理位置,而這個(gè)位置是可以賦給一個(gè)指針的。一零點(diǎn)函數(shù)的地址就是該函數(shù)的入口點(diǎn)。因此,函數(shù)指針可被用來調(diào)用一個(gè)函數(shù)。函數(shù)的地址是用不帶任何括號或參數(shù)的函數(shù)名來得到的。(這很類似于數(shù)組地址的得到方法,即,在只有數(shù)組名而無下標(biāo)是就得到數(shù)組地址。)
網(wǎng)站建設(shè)哪家好,找創(chuàng)新互聯(lián)!專注于網(wǎng)頁設(shè)計(jì)、網(wǎng)站建設(shè)、微信開發(fā)、微信小程序開發(fā)、集團(tuán)企業(yè)網(wǎng)站建設(shè)等服務(wù)項(xiàng)目。為回饋新老客戶創(chuàng)新互聯(lián)還提供了麻城免費(fèi)建站歡迎大家使用!
怎樣說明一個(gè)函數(shù)指針變量呢 ?
為了說明一個(gè)變量 fn_pointer 的類型是"返回值為 int 的函數(shù)指針", 你可以使用下面的說明語句:
int (*fn_pointer) ();
為了讓編譯器能正確地解釋這句語句, *fn_pointer 必須用括號圍起來。若漏了這對括號, 則:
int *fn_pointer ();
的意思完全不同了。fn_pointer 將是一個(gè)函數(shù)名, 其返回值為 int 類型的指針。
2:函數(shù)指針變量
在C語言中規(guī)定,一個(gè)函數(shù)總是占用一段連續(xù)的內(nèi)存區(qū), 而函數(shù)名就是該函數(shù)所占內(nèi)存區(qū)的首地址。 我們可以把函數(shù)的這個(gè)首地址 ( 或稱入口地址 ) 賦予一個(gè)指針變量, 使該指針變量指向該函數(shù)。然后通過指針變量就可以找到并調(diào)用這個(gè)函數(shù)。我們把這種指向函數(shù)的指針變量稱為 " 函數(shù)指針變量 " 。
函數(shù)指針變量定義的一般形式為:
類型說明符 (* 指針變量名 )();
其中 " 類型說明符 " 表示被指函數(shù)的返回值的類型。 "(* 指針變量名 )" 表示 "*" 后面的變量是定義的指針變量。 最后的空括號表示指針變量所指的是一個(gè)函數(shù)。
例如: int (*pf)();
表示 pf 是一個(gè)指向函數(shù)入口的指針變量,該函數(shù)的返回值 ( 函數(shù)值 ) 是整型。
下面通過例子來說明用指針形式實(shí)現(xiàn)對函數(shù)調(diào)用的方法。
int max(int a,int b)
{
if(ab)return a;
else return b;
}
main()
{
int max(int a,int b);
int(*pmax)();
int x,y,z;
pmax=max;
printf("input two numbers:/n");
scanf("%d%d",x,y);
z=(*pmax)(x,y);
printf("maxmum=%d",z);
}
從上述程序可以看出用,函數(shù)指針變量形式調(diào)用函數(shù)的步驟如下:
1. 先定義函數(shù)指針變量,如后一程序中第 9 行 int (*pmax)(); 定義 pmax 為函數(shù)指針變量。
2. 把被調(diào)函數(shù)的入口地址 ( 函數(shù)名 ) 賦予該函數(shù)指針變量,如程序中第 11 行 pmax=max;
3. 用函數(shù)指針變量形式調(diào)用函數(shù),如程序第 14 行 z=(*pmax)(x,y); 調(diào)用函數(shù)的一般形式為: (* 指針變量名 ) ( 實(shí)參表 ) 使用函數(shù)指針變量還應(yīng)注意以下兩點(diǎn):
a. 函數(shù)指針變量不能進(jìn)行算術(shù)運(yùn)算,這是與數(shù)組指針變量不同的。數(shù)組指針變量加減一個(gè)整數(shù)可使指針移動(dòng)指向后面或前面的數(shù)組元素,而函數(shù)指針的移動(dòng)是毫無意義的。
b. 函數(shù)調(diào)用中 "(* 指針變量名 )" 的兩邊的括號不可少,其中的 * 不應(yīng)該理解為求值運(yùn)算,在此處它只是一種表示符號。
3:指針型函數(shù)
前面我們介紹過,所謂函數(shù)類型是指函數(shù)返回值的類型。 在C語言中允許一個(gè)函數(shù)的返回值是一個(gè)指針 ( 即地址 ) ,這種返回指針值的函數(shù)稱為指針型函數(shù)。
定義指針型函數(shù)的一般形式為:
類型說明符 * 函數(shù)名 ( 形參表 )
{
…… /* 函數(shù)體 */
}
其中函數(shù)名之前加了 "*" 號表明這是一個(gè)指針型函數(shù),即返回值是一個(gè)指針。類型說明符表示了返回的指針值所指向的數(shù)據(jù)類型。
如:
int *ap(int x,int y)
{
…… /* 函數(shù)體 */
}
表示 ap 是一個(gè)返回指針值的指針型函數(shù), 它返回的指針指向一個(gè)整型變量。下例中定義了一個(gè)指針型函數(shù) day_name ,它的返回值指向一個(gè)字符串。該函數(shù)中定義了一個(gè)靜態(tài)指針數(shù)組 name 。 name 數(shù)組初始化賦值為八個(gè)字符串,分別表示各個(gè)星期名及出錯(cuò)提示。形參 n 表示與星期名所對應(yīng)的整數(shù)。在主函數(shù)中, 把輸入的整數(shù) i 作為實(shí)參, 在 printf 語句中調(diào)用 day_name 函數(shù)并把 i 值傳送給形參 n 。 day_name 函數(shù)中的 return 語句包含一個(gè)條件表達(dá)式, n 值若大于 7 或小于 1 則把 name[0] 指針返回主函數(shù)輸出出錯(cuò)提示字符串 "Illegal day" 。否則返回主函數(shù)輸出對應(yīng)的星期名。主函數(shù)中的第 7 行是個(gè)條件語句,其語義是,如輸入為負(fù)數(shù) (i0) 則中止程序運(yùn)行退出程序。 exit 是一個(gè)庫函數(shù), exit(1) 表示發(fā)生錯(cuò)誤后退出程序, exit(0) 表示正常退出。
應(yīng)該特別注意的是函數(shù)指針變量和指針型函數(shù)這兩者在寫法和意義上的區(qū)別。如 int(*p)() 和 int *p() 是兩個(gè)完全不同的量。 int(*p)() 是一個(gè)變量說明,說明 p 是一個(gè)指向函數(shù)入口的指針變量,該函數(shù)的返回值是整型量, (*p) 的兩邊的括號不能少。
int *p() 則不是變量說明而是函數(shù)說明,說明 p 是一個(gè)指針型函數(shù),其返回值是一個(gè)指向整型量的指針,*p 兩邊沒有括號。作為函數(shù)說明, 在括號內(nèi)最好寫入形式參數(shù),這樣便于與變量說明區(qū)別。 對于指針型函數(shù)定義,int *p() 只是函數(shù)頭部分,一般還應(yīng)該有函數(shù)體部分。
main()
{
int i;
char *day_name(int n);
printf("input Day No:/n");
scanf("%d",i);
if(i0) exit(1);
printf("Day No:%2d--%s/n",i,day_name(i));
}
char *day_n
ame(int n)
{
static char *name[]={ "Illegal day",
"Monday",
"Tuesday",
"Wednesday",
"Thursday",
"Friday",
"Saturday",
"Sunday"};
return((n1||n7) ? name[0] : name[n]);
}
本程序是通過指針函數(shù),輸入一個(gè) 1 ~ 7 之間的整數(shù), 輸出對應(yīng)的星期名。指針數(shù)組的說明與使用一個(gè)數(shù)組的元素值為指針則是指針數(shù)組。指針數(shù)組是一組有序的指針的集合。指針數(shù)組的所有元素都必須是具有相同存儲(chǔ)類型和指向相同數(shù)據(jù)類型的指針變量。
指針數(shù)組說明的一般形式為: 類型說明符 * 數(shù)組名 [ 數(shù)組長度 ]
其中類型說明符為指針值所指向的變量的類型。例如: int *pa[3] 表示 pa 是一個(gè)指針數(shù)組,它有三個(gè)數(shù)組元素, 每個(gè)元素值都是一個(gè)指針,指向整型變量。通??捎靡粋€(gè)指針數(shù)組來指向一個(gè)二維數(shù)組。 指針數(shù)組中的每個(gè)元素被賦予二維數(shù)組每一行的首地址,因此也可理解為指向一個(gè)一維數(shù)組。圖 6—6 表示了這種關(guān)系。
int a[3][3]={1,2,3,4,5,6,7,8,9};
int *pa[3]={a[0],a[1],a[2]};
int *p=a[0];
main()
{
int i;
for(i=0;i3;i++)
printf("%d,%d,%d/n",a[i][2-i],*a[i],*(*(a+i)+i));
for(i=0;i3;i++)
printf("%d,%d,%d/n",*pa[i],p[i],*(p+i));
}
本例程序中, pa 是一個(gè)指針數(shù)組,三個(gè)元素分別指向二維數(shù)組 a 的各行。然后用循環(huán)語句輸出指定的數(shù)組元素。其中 *a[i] 表示 i 行 0 列元素值; *(*(a+i)+i) 表示 i 行 i 列的元素值; *pa[i] 表示 i 行 0 列元素值;由于 p 與 a[0] 相同,故 p[i] 表示 0 行 i 列的值; *(p+i) 表示 0 行 i 列的值。讀者可仔細(xì)領(lǐng)會(huì)元素值的各種不同的表示方法。 應(yīng)該注意指針數(shù)組和二維數(shù)組指針變量的區(qū)別。 這兩者雖然都可用來表示二維數(shù)組,但是其表示方法和意義是不同的
有函數(shù): int fun(int a,int b);\x0d\x0a要定義指向該函數(shù)的指針\x0d\x0a對比指向 int a; 的指針\x0d\x0aint *p; p = a;\x0d\x0ap的定義是怎么來的?\x0d\x0a首先要保證p是一個(gè)指針類型\x0d\x0a寫下(*p),\x0d\x0a然后,考慮下p的基類型,\x0d\x0ap的基類型就是變量a的類型int\x0d\x0a將int 放在(*p)前面就行了\x0d\x0aint (*p); \x0d\x0a括號可以省略,就成了 int *p;\x0d\x0a\x0d\x0a同理\x0d\x0a想要實(shí)現(xiàn) pf = fun;\x0d\x0a(*pf) 將pf定義為一個(gè)指針,\x0d\x0a將fun的類型作為pf的基類型\x0d\x0afun相當(dāng)于一個(gè) int (int a,int b)類型的量\x0d\x0aint (int a,int b) (*pf);\x0d\x0a基類型中有圓括號和中括號要后移\x0d\x0aint (*pf)(int a,int b) ;//括號不能省略\x0d\x0apf = fun;\x0d\x0a調(diào)用時(shí)\x0d\x0a(*pf)(3,4); pf(3,4)都可以
這個(gè)其實(shí)很好理解
只需要把%s,%c,%p放在一起看
首先%s是最常見的很好理解,就是告訴程序后面的變量是一個(gè)字符串,在C語言中也就是字符數(shù)組,類型是char*或者char[],換句話說,你告訴程序我會(huì)給你一個(gè)指針,你去讀這個(gè)指針指向的值。
然后%c是告訴程序后面的變量是個(gè)字符,類型是char,差別已經(jīng)很明顯了,這時(shí)候你告訴程序我給你提供一個(gè)字符,你給我在%c處打印出來。然而你給的卻是一個(gè)指針,那程序就直接把你給的指針指向的地址本身當(dāng)成字符打印出來了。但是一個(gè)地址肯定不止一個(gè)字節(jié),也就是說超過了char應(yīng)該有的大小,這時(shí)候程序會(huì)直接忽略了超出大小的部分,只讀第一個(gè)字節(jié)。
如果你同時(shí)使用%p,告訴程序,你會(huì)提供一個(gè)指針,直接把這個(gè)指針指向的地址給我打印出來。把輸出的地址最后兩位的16進(jìn)制數(shù)查ascii表,換算成字符,你會(huì)發(fā)現(xiàn),剛好就是前面%c打印出的字符。
函數(shù)指針告訴CPU下一步需要執(zhí)行哪里的代碼的指針, 一數(shù)組十分相像.
#include?stdio.h
void?function(int?i,?double?d)????//一個(gè)函數(shù)的聲明
{
printf("函數(shù)運(yùn)行,?%d,?%lf\n",?i,?d);
}
void?AnotherFunction(int?i,?double?d)????//另外一個(gè)函數(shù)聲明
{
printf("又一個(gè)函數(shù)運(yùn)行,?%d,?%lf\n",?i,?d);
}
int?main()
{
void?(*pFun)(int,?double);????//定義了一個(gè)函數(shù)指針
int?(*p)(int?,?double);????//用于調(diào)試的函數(shù)指針
pFun?=?function;????//讓這個(gè)函數(shù)指針指向第一個(gè)函數(shù)
pFun(10,?10.101);????//通過這個(gè)函數(shù)指針來調(diào)用第一個(gè)函數(shù)
pFun?=?AnotherFunction;????//讓這個(gè)函數(shù)指針指向第二個(gè)函數(shù)
pFun(20,?20.202);????//通過這個(gè)函數(shù)指針來調(diào)用第二個(gè)函數(shù)
//若將以下代碼取消注釋,?就會(huì)產(chǎn)生錯(cuò)誤
/*
p?=?function;
p?=?AnotherFunction;
//原因很簡單,?因?yàn)閜是必須指向一個(gè)返回值為int,?第一個(gè)參數(shù)為int,?第二個(gè)參數(shù)為double的函數(shù),?但是在此代碼中定義的函數(shù)返回值是void
*/
//這個(gè)實(shí)驗(yàn)中,?直接用函數(shù)名來賦值了,?是因?yàn)楹瘮?shù)名其實(shí)就是這個(gè)函數(shù)的指針
//而在平常調(diào)用函數(shù)時(shí),?都要加上(),?如:printf("");
//這里,?"()"就相當(dāng)于'*'(就是取值運(yùn)算符),?printf是函數(shù)指針,?就相當(dāng)于是數(shù)組名,?即數(shù)組的首地址
return?0;
}
網(wǎng)頁名稱:c語言查表函數(shù)指針,c語言函數(shù)速查手冊
網(wǎng)頁路徑:http://www.rwnh.cn/article22/dsijojc.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供標(biāo)簽優(yōu)化、App設(shè)計(jì)、網(wǎng)站維護(hù)、搜索引擎優(yōu)化、網(wǎng)站改版、網(wǎng)站內(nèi)鏈
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)