1、指針函數:
指針函數是指帶指針的函數,即本質是一個函數。我們知道函數都有返回類型(如果不返回值,則為無值型),只不過指針函數返回類型是某一類型的指針。其定義格式如下所示:
返回類型標識符 *返回名稱(形式參數表)
{ 函數體 }
返回類型可以是任何基本類型和復合類型。返回指針的函數的用途十分廣泛。事實上,每一個函數,即使它不帶有返回某種類型的指針,它本身都有一個入口地址,該地址相當於一個指針。比如函數返回一個整型值,實際上也相當於返回一個指針變量的值,不過這時的變量是函數本身而已,而整個函數相當於一個"變量"。例如下面一個返回指針函數的例子:
float *find(float (*pionter)[4], int n) /* 定義指針函數 */ { float *pt; pt = *(pionter + n); return(pt); } main() { static float score[][4] = {{60, 70, 80, 90}, {56, 89, 34, 45}, {34, 23, 56, 45}}; float *p; int i, m; printf("Enter the number to be found:"); scanf("%d", &m); printf("the score of NO.%d are:\n", m); p = find(score, m); for (i = 0; i < 4; i++) { printf("%5.2f\t", *(p + i)); } }
學生學號從0號算起,函數find()被定義為指針函數,起形參pointer是指針指向包含4個元素的一維數組的指針變量。pointer + 1指向score的第一行。*(pointer + 1)指向第一行的第0個元素。pt是一個指針變量,它指向浮點型變量。main()函數中調用find()函數,將score數組的首地址傳給pointer.
將字符串1(str1)連接字符串2(str2),並輸出字符串1.
#include "stdio.h" char *mystrcpy(char *str1, char *str2) { char *p; p = str1; while (*str1) { str1++; } while (*str1++ = *str2++); return p; } int main(void) { char str1[] = "I LOVE SHY"; char str2[] = " chj!"; char *p; p = mystrcpy(str1, str2); printf("%s\n", p); }
例3:
int *GetDate(int wk, int dy) { static int calendar[5][7] = { {1, 2, 3, 4, 5, 6, 7}, {8, 9, 10, 11, 12, 13, 14}, {15, 16, 17, 18, 19, 20, 21}, {22, 23, 24, 25, 26, 27, 28}, {29, 30, 31, -1, 0 } }; return (&calendar[wk-1][dy-1]); } int main(void) { int wk, dy; do { printf("Enter week(1-5)day(1-7)\n"); scanf("%d %d", &wk, &dy); } while (wk < 1 || wk > 5 || dy < 1 || dy > 7); printf("%d", *GetDate(wk, dy)); }
2、函數指針:
"函數指針"是指向函數的指針變量,因而"函數指針"本身首先應是指針變量,只不過該指針變量指向函數。這正如用指針變量可指向整型變量、字符型、數組一樣,這里是指向函數。如前所述,C在編譯時,每一個函數都有一個入口地址,該入口地址就是函數指針所指向的地址。有了指向函數的指針變量后,可用該指針變量調用函數,就如同用指針變量可引用其他類型變量一樣,在這些概念上一致的。函數指針有兩個用途:調用函數和做函數的參數。函數指針的說明方法為:
函數類型 (*指針變量名)(形參列表);
"函數類型"說明函數的返回類型,由於"()"的優先級高於"*", 所以指針變量名外的括號必不可少,后面的"形參列表"表示指針變量指向的函數所帶的參數列表。
例如:
int (*f)(int x);
double(*ptr)(double x);
在定義函數指針時請注意:
函數指針和它指向的函數的參數個數和類型都應該是—致的;
函數指針的類型和函數的返回值類型也必須是一致的。
函數指針的賦值
函數名和數組名一樣代表了函數代碼的首地址,因此在賦值時,直接將函數指針指向函數名就行了。
例如,
int func(int x); /* 聲明一個函數 */ int (*f)(int x); /* 聲明一個函數指針 */ f = func; /* 將func函數的首地址賦給指針f */
賦值時函數func不帶括號,也不帶參數,由於func代表函數的首地址,因此經過賦值以后,指針f就指向函數func(x)的代碼的首地址。
與其他指針變量相類似,如果指針變量pi是指向某整型變量i的指針,則*p等於它所指的變量i;如果pf是指向某浮點型變量f的指針,則*pf就等價於它所指的變量f。同樣地,*f是指向函數func(x)的指針,則*f就代表它所指向的函數func。所以在執行了f = func;
之后,(*f)和func代表同一函數。
由於函數指針指向存儲區中的某個函數,因此可以通過函數指針調用相應的函數。現在我們就討論如何用函數指針調用函數,它應執行下面三步:
首先,要說明函數指針變量。
例如:int (*f)(int x);
其次,要對函數指針變量賦值。
例如: f = func;
(func(x)必須先要有定義)
最后,要用(*指針變量)(參數表);
調用函數。
例如:(*f)(x);
(x必須先賦值)
例1:
int max(int x, int y) { return(x > y ? x : y); } int main(void) { int a, b, c; int (*ptr)(int, int); scanf("%d,%d", &a, &b); ptr = max; c = (*ptr)(a, b); printf("a=%d,b=%d,max=%d", a, b, c); return 0; }
例2:
void FileFunc() { printf("FileFunc\n"); } void EditFunc() { printf("EditFunc\n"); } int main(void) { void (*funcp)(); funcp = FileFunc; (*funcp)(); funcp = EditFunc; (*funcp)(); }
ptr是指向函數的指針變量,所以可把函數max()賦給ptr作為ptr的值,即把max()的入口地址賦給ptr, 以后就可以用ptr來調用該函數,實際上ptr和max都指向同一個入口地址,不同就是ptr是一個指針變量,不像函數名稱那樣是死的,它可以指向任何函數,就看你像怎么做了。在程序中把哪個函數的地址賦給它,它就指向哪個函數。而后用指針變量調用它,因此可以先后指向不同的函數,不過注意,指向函數的指針變量沒有++和--運算,用時要小心。
3、函數指針數組
關於函數指針數組的定義
關於函數指針數組的定義方法,有兩種:一種是標准的方法;一種是蒙騙法。
第一種,標准方法:
{
分析:函數指針數組是一個其元素是函數指針的數組。那么也就是說,此數據結構是是一個數組,且其元素是一個指向函數入口地址的指針。
根據分析:首先說明是一個數組:數組名[]
其次,要說明其元素的數據類型指針:
*數組名[].
再次,要明確這每一個數組元素是指向函數入口地址的指針:函數返回值類型(*數組名[])().請注意,這里為什么要把"*數組名[]"用括號擴起來呢?因為圓括號和數組說明符的優先級是等同的,如果不用圓括號把指針數組說明表達式擴起來,根據圓括號和方括號的結合方向,那么 *數組名[]() 說明的是什么呢?是元素返回值類型為指針的函數數組。有這樣的函數數祖嗎?不知道。所以必須括起來,以保證數組的每一個元素是指針。
}
第二種,蒙騙法:
盡管函數不是變量,但它在內存中仍有其物理地址,該地址能夠賦給指針變量。獲取函數方法是:用不帶有括號和參數的函數名得到。
函數名相當於一個指向其函數入口指針常量。 那么既然函數名是一個指針常量,那么就可以對其進行一些相應的處理,如強制類型轉換。
那么我們就可以把這個地址放在一個整形指針數組中,然后作為函數指針調用即可。
完整例子:
#include "stdio.h" int add1(int a1, int b1); int add2(int a2, int b2); int main(int argc, char *argv[]) { int numa1 = 1, numb1 = 2; int numa2 = 2, numb2 = 3; int (*op[2])(int a, int b); op[0] = add1; op[1] = add2; printf("%d %d\n", op[0](numa1, numb1), op[1](numa2, numb2)); getch(); } int add1(int a1, int b1) { return a1 + b1; } int add2(int a2, int b2) { return a2 + b2; }
再給出常用的C變量的定義方式:
a) 一個整型數(An integer)
b) 一個指向整型數的指針(A pointer to an integer)
c) 一個指向指針的的指針,它指向的指針是指向一個整型數(A pointer to a pointer to an integer)
d) 一個有10個整型數的數組(An array of 10 integers)
e) 一個有10個指針的數組,該指針是指向一個整型數的(An array of 10 pointers to integers)
f) 一個指向有10個整型數數組的指針(A pointer to an array of 10 integers)
g) 一個指向函數的指針,該函數有一個整型參數並返回一個整型數(A pointer to a function that takes an integer as an argument and returns an integer)
h) 一個有10個指針的數組,該指針指向一個函數,該函數有一個整型參數並返回一個整型數( An array of ten pointers to functions that take an integer argument and return an integer)
答案是:
a) int a; // An integer
b) int *a; // A pointer to an integer
c) int **a; // A pointer to a pointer to an integer
d) int a[10]; // An array of 10 integers
e) int *a[10]; // An array of 10 pointers to integers
f) int (*a)[10]; // A pointer to an array of 10 integers
g) int (*a)(int); // A pointer to a function a that takes an integer argument and returns an integer
h) int (*a[10])(int); // An array of 10 pointers to functions that take an integer argument and return an integer