c_lang_Reference_040820_04


文字來源 VC世界--C語言教室

自我存檔,也獻給和我一樣的DElphi程序員


函數指針變量

  在C語言中規定,一個函數總是佔用一段連續的內存區, 而函數名就是該函數所佔內存區的首地址。
函數指針變量定義的一般形式爲:
類型說明符 (*指針變量名)();
其中“類型說明符”表示被指函數的返回值的類型。“(* 指針變量名)”表示“*”後面的變量是定義的指針變量。
例如: int (*pf)();
表示pf是一個指向函數入口的指針變量,該函數的返回值(函數值)是整型。
下面通過例子來說明用指針形式實現對函數調用的方法。
int max(int a,int b){
if(a>b)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);
}

指針型函數

前面我們介紹過,所謂函數類型是指函數返回值的類型。 在C語言中允許一個函數的返回值是一個指針(即地址), 這種返回指針值的函數稱爲指針型函數。
定義指針型函數的一般形式爲:
類型說明符 *函數名(形參表)
{
…… /*函數體*/
}

如:
int *ap(int x,int y)
{
...... /*函數體*/
}

main(){
int i;
char *day_name(int n);
printf("input Day No:/n");
scanf("%d",&i);
if(i<0) exit(1);
printf("Day No:%2d-->%s/n",i,day_name(i));
}
char *day_name(int n){
static char *name[]={ "Illegal day",
"Monday",
"Tuesday",
"Wednesday",
"Thursday",
"Friday",
"Saturday",
"Sunday"};
return((n<1||n>7) ? name[0] : name[n]);
}

  指針數組說明的一般形式爲: 類型說明符*數組名[數組長度]
  其中類型說明符爲指針值所指向的變量的類型。例如: int *pa[3] 表示pa是一個指針數組,它有三個數組元素, 每個元素值都是一個指針,指向整型變量。通常可用一個指針數組來指向一個二維數組。 指針數組中的每個元素被賦予二維數組每一行的首地址, 因此也可理解爲指向一個一維數組。圖6—6表示了這種關係。
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;i<3;i++)
printf("%d,%d,%d/n",a[i][2-i],*a[i],*(*(a+i)+i));
for(i=0;i<3;i++)
printf("%d,%d,%d/n",*pa[i],p[i],*(p+i));
}
  本例程序中,pa是一個指針數組,三個元素分別指向二維數組a的各行。然後用循環語句輸出指定的數組元素。其中*a[i]表示i行0列元素值;*(*(a+i)+i)表示i行i列的元素值;*pa[i]表示i行0列元素值;由於p與a[0]相同,故p[i]表示0行i列的值;*(p+i)表示0行i列的值。讀者可仔細領會元素值的各種不同的表示方法。 應該注意指針數組和二維數組指針變量的區別。 這兩者雖然都可用來表示二維數組,但是其表示方法和意義是不同的。

  二維數組指針變量是單個的變量,其一般形式中"(*指針變量名)"兩邊的括號不可少。而指針數組類型表示的是多個指針( 一組有序指針)在一般形式中"*指針數組名"兩邊不能有括號。例如: int (*p)[3];表示一個指向二維數組的指針變量。該二維數組的列數爲3或分解爲一維數組的長度爲3。 int *p[3] 表示p是一個指針數組,有三個下標變量p[0],p[1],p[2]均爲指針變量。

  指針數組也常用來表示一組字符串, 這時指針數組的每個元素被賦予一個字符串的首地址。 指向字符串的指針數組的初始化更爲簡單。例如在例6.20中即採用指針數組來表示一組字符串。 其初始化賦值爲:
char *name[]={"Illagal day",
"Monday",
"Tuesday",
"Wednesday",
"Thursday",
"Friday",
"Saturday",
"Sunday"};
  完成這個初始化賦值之後,name[0]即指向字符串"Illegal day",name[1]指?quot;Monday"......。

  指針數組也可以用作函數參數。在本例主函數中,定義了一個指針數組name,並對name 作了初始化賦值。其每個元素都指向一個字符串。然後又以name 作爲實參調用指針型函數day name,在調用時把數組名 name 賦予形參變量name,輸入的整數i作爲第二個實參賦予形參n。在day name函數中定義了兩個指針變量pp1和pp2,pp1被賦予name[0]的值(即*name),pp2被賦予name[n]的值即*(name+ n)。由條件表達式決定返回pp1或pp2指針給主函數中的指針變量ps。最後輸出i和ps的值。

指針數組作指針型函數的參數
main(){
static char *name[]={ "Illegal day",
"Monday",
"Tuesday",
"Wednesday",
"Thursday",
"Friday",
"Saturday",
"Sunday"};
char *ps;
int i;
char *day name(char *name[],int n);
printf("input Day No:/n");
scanf("%d",&i);
if(i<0) exit(1);
ps=day name(name,i);
printf("Day No:%2d-->%s/n",i,ps);
}
char *day name(char *name[],int n)
{
char *pp1,*pp2;
pp1=*name;
pp2=*(name+n);
return((n<1||n>7)? pp1:pp2);
}
下例要求輸入5個國名並按字母順序排列後輸出。
#include"string.h"
main(){
void sort(char *name[],int n);
void print(char *name[],int n);
static char *name[]={ "CHINA","AMERICA","AUSTRALIA",
"FRANCE","GERMAN"};
int n=5;
sort(name,n);
print(name,n);
}
void sort(char *name[],int n){
char *pt;
int i,j,k;
for(i=0;i<n-1;i++){
k=i;
for(j=i+1;j<n;j++)
if(strcmp(name[k],name[j])>0) k=j;
if(k!=i){
pt=name[i];
name[i]=name[k];
name[k]=pt;
}
}
}
void print(char *name[],int n){
int i;
for (i=0;i<n;i++) printf("%s/n",name[i]);
}

main函數的參數

  前面介紹的main函數都是不帶參數的。因此main 後的括號都是空括號。實際上,main函數可以帶參數,這個參數可以認爲是 main函數的形式參數。C語言規定main函數的參數只能有兩個, 習慣上這兩個參數寫爲argc和argv。因此,main函數的函數頭可寫爲: main (argc,argv)C語言還規定argc(第一個形參)必須是整型變量,argv( 第二個形參)必須是指向字符串的指針數組。加上形參說明後,main函數的函數頭應寫爲:
main (argc,argv)
int argv;
char *argv[];或寫成:
main (int argc,char *argv[])
  由於main函數不能被其它函數調用, 因此不可能在程序內部取得實際值。那麼,在何處把實參值賦予main函數的形參呢? 實際上,main函數的參數值是從操作系統命令行上獲得的。當我們要運行一個可執行文件時,在DOS提示符下鍵入文件名,再輸入實際參數即可把這些實參傳送到main的形參中去。

  DOS提示符下命令行的一般形式爲: C:/>可執行文件名 參數 參數……; 但是應該特別注意的是,main 的兩個形參和命令行中的參數在
位置上不是一一對應的。因爲,main的形參只有二個,而命令行中的參數個數原則上未加限制。argc參數表示了命令行中參數的個數(注意:文件名本身也算一個參數),argc的值是在輸入命令行時由系統按實際參數的個數自動賦予的。例如有命令行爲: C:/>E6 24 BASIC dbase FORTRAN由於文件名E6 24本身也算一個參數,所以共有4個參數,因此argc取得的值爲4。argv參數是字符串指針數組,其各元素值爲命令行中各字符串(參數均按字符串處理)的首地址。 指針數組的長度即爲參數個數。數組元素初值由系統自動賦予。其表示如圖6.8所示:
main(int argc,char *argv){
while(argc-->1)
printf("%s/n",*++argv);
}
本例是顯示命令行中輸入的參數如果上例的可執行文件名爲e24.exe,存放在A驅動器的盤內。
因此輸入的命令行爲: C:/>a:e24 BASIC dBASE FORTRAN
則運行結果爲:
BASIC
dBASE
FORTRAN
  該行共有4個參數,執行main時,argc的初值即爲4。argv的4個元素分爲4個字符串的首地址。執行while語句,每循環一次 argv值減1,當argv等於1時停止循環,共循環三次, 因此共可輸出三個參數。在printf函數中,由於打印項*++argv是先加1再打印, 故第一次打印的是argv[1]所指的字符串BASIC。第二、 三次循環分別打印後二個字符串。而參數e24是文件名,不必輸出。

  下例的命令行中有兩個參數,第二個參數20即爲輸入的n值。在程序中*++argv的值爲字符串“20”,然後用函數"atoi"把它換爲整型作爲while語句中的循環控制變量,輸出20個偶數。
#include"stdlib.h"
main(int argc,char*argv[]){
int a=0,n;
n=atoi(*++argv);
while(n--) printf("%d ",a++*2);
}
  本程序是從0開始輸出n個偶數。指向指針的指針變量如果一個指針變量存放的又是另一個指針變量的地址, 則稱這個指針變量爲指向指針的指針變量。

  在前面已經介紹過,通過指針訪問變量稱爲間接訪問, 簡稱間訪。由於指針變量直接指向變量,所以稱爲單級間訪。 而如果通過指向指針的指針變量來訪問變量則構成了二級或多級間訪。在C語言程序中,對間訪的級數並未明確限制, 但是間訪級數太多時不容易理解解,也容易出錯,因此,一般很少超過二級間訪。 指向指針的指針變量說明的一般形式爲:
類型說明符** 指針變量名;
例如: int ** pp; 表示pp是一個指針變量,它指向另一個指針變量, 而這個指針變量指向一個整型量。下面舉一個例子來說明這種關係。
main(){
int x,*p,**pp;
x=10;
p=&x;
pp=&p;
printf("x=%d/n",**pp);
}
  上例程序中p 是一個指針變量,指向整型量x;pp也是一個指針變量, 它指向指針變量p。通過pp變量訪問x的寫法是**pp。程序最後輸出x的值爲10。通過上例,讀者可以學習指向指針的指針變量的說明和使用方法。

  下述程序中首先定義說明了指針數組ps並作了初始化賦值。 又說明了pps是一個指向指針的指針變量。在5次循環中, pps 分別取得了ps[0],ps[1],ps[2],ps[3],ps[4]的地址值(如圖6.10所示)。再通過這些地址即可找到該字符串。
main(){
static char *ps[]={ "BASIC","DBASE","C","FORTRAN",
"PASCAL"};
char **pps;
int i;
for(i=0;i<5;i++){
pps=ps+i;
printf("%s/n",*pps);
}
}
本程序是用指向指針的指針變量編程,輸出多個字符串。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章