深入理解C C 函式指標

2021-04-14 09:11:36 字數 3578 閱讀 7027

函式指標陣列的妙用

筆者在開發某軟體過程中遇到這樣乙個問題,前級模組傳給我二進位制資料,輸入引數為 char* buffer和 int length,buffer是資料的首位址,length表示這批資料的長度。資料的特點是:長度不定,型別不定,由第乙個位元組(buffer[0])標識該資料的型別,共有256(28 )種可能性。我的任務是必須對每一種可能出現的資料型別都要作處理,並且我的模組包含若干個函式,在每個函式裡面都要作類似的處理。若按通常做法,會寫出如下**:

void myfuntion( char* buffer, int length )

}

如果按照這種方法寫下去,那麼在我的每乙個函式裡面,都必須作如此多的判斷,寫出的**肯定很長,並且每一次處理,都要作許多次判斷之後才找到正確的處理函式,**的執行效率也不高。針對上述問題,我想到了用函式指標陣列的方法解決這個問題。

函式指標的概念,在潭浩強先生的c語言程式設計這本經典的教程中提及過,在大多數情況下我們使用不到,也忽略了它的存在。函式名實際上也是一種指標,指向函式的入口位址,但它又不同於普通的如int*、double*指標,看下面的例子來理解函式指標的概念:

int funtion( int x, int y );

void main ( void )

語句1定義了乙個函式function,其輸入為兩個整型數,返回也為乙個整型數(輸入引數和返回值可為其它任何資料型別);語句3定義了乙個函式指標,與int*或double*定義指標不同的是,函式指標的定義必須同時指出輸入引數,表明這是乙個函式指標,並且*fun也必須用一對括號括起來;語句6將函式指標賦值為funtion,前提條件是*fun和function的輸入引數和返回值必須保持一致。語句5直接呼叫函式function(),語句7是呼叫函式指標,二者等效。

當然從上述例子看不出函式指標的優點,目的主要是想引出函式指標陣列的概念。我們從上面例子可以得知,既然函式名可以通過函式指標加以儲存,那們也一定能定義乙個陣列儲存若干個函式名,這就是函式指標陣列。正確使用函式指標陣列的前提條件是,這若干個需要通過函式指標陣列儲存的函式必須有相同的輸入、輸出值。

這樣,我工作中所面臨的問題可以解決如下:

首先定義256個處理函式(及其實現)。

void funtion0( void );

……void funtion255(void );

其次定義函式指標陣列,並給陣列賦值。

void (*fun[256])(void);

fun[0] = function0;

……fun[255] = function();

最後,myfunction()函式可以修改如下:

void myfuntion( char* buffer, int length )

只要2行**,就完成了256條case語句要做的事,減少了編寫**時工作量,將nstreamtype作為陣列下標,直接呼叫函式指標,從**執行效率上來說,也比case語句高。假如多個函式中均要作如此處理,函式指標陣列更能體現出它的優勢。

函式指標與typedef

關於c++中函式指標的使用(包含對typedef用法的討論)

(一)簡單的函式指標的應用。

//形式1:返回型別(*函式名)(參數列)

char (*pfun)(int);

char glfun(int a)

void main()

第一行定義了乙個指標變數pfun。首先我們根據前面提到的「形式1」認識到它是乙個指向某種函式的指標,這種函式引數是乙個int型,返回值是char型別。只有第一句我們還無法使用這個指標,因為我們還未對它進行賦值。

第二行定義了乙個函式glfun()。該函式正好是乙個以int為引數返回char的函式。我們要從指標的層次上理解函式——函式的函式名實際上就是乙個指標,函式名指向該函式的**在記憶體中的首位址。

然後就是可愛的main()函式了,它的第一句您應該看得懂了——它將函式glfun的位址賦值給變數pfun。main()函式的第二句中「*pfun」顯然是取pfun所指向位址的內容,當然也就是取出了函式glfun()的內容,然後給定引數為2。

(二)使用typedef更直觀更方便。

//形式2:typedef 返回型別(*新型別)(參數列)

typedef char (*ptrfun)(int);

ptrfun pfun;

char glfun(int a)

void main()

typedef的功能是定義新的型別。第一句就是定義了一種ptrfun的型別,並定義這種型別為指向某種函式的指標,這種函式以乙個int為引數並返回char型別。後面就可以像使用int,char一樣使用ptrfun了。

第二行的**便使用這個新型別定義了變數pfun,此時就可以像使用形式1一樣使用這個變數了。

(三)在c++類中使用函式指標。

//形式3:typedef 返回型別(類名::*新型別)(參數列)

class ca

};ca ca;

typedef char (ca::*ptrfun)(int);

ptrfun pfun;

void main()

在這裡,指標的定義與使用都加上了「類限制」或「物件」,用來指明指標指向的函式是哪個類的,這裡的類物件也可以是使用new得到的。比如:

ca *pca = new ca;

pca->(*pfun)(2);

delete pca;

而且這個類物件指標可以是類內部成員變數,你甚至可以使用this指標。比如:

類ca有成員變數ptrfun m_pfun;

void ca::lcfun2()

一句話,使用類成員函式指標必須有「->*」或「.*」的呼叫。

int   docase(int, long);

則,在呼叫動態庫是有兩種方法:

1.  先宣告乙個與動態庫中型別一致的指標函式變數:

int (*docase)(int ,long);//

用於指向動態庫中的docase函式位址

hinstance glibmydll = null;

glibmydll = loadlibrary("test.dll");

if(glibmydll != null)

//呼叫函式

int s = docase(1,1000);

2.用typedef定義乙個指標函式:typedef (*docase)(int ,long);

hinstance glibmydll = null;

docase _docase;

glibmydll = loadlibrary("test.dll");

if(glibmydll != null)

//呼叫函式

深入理解C C 函式指標

函式指標陣列的妙用 筆者在開發某軟體過程中遇到這樣乙個問題,前級模組傳給我二進位制資料,輸入引數為 char buffer和 int length,buffer是資料的首位址,length表示這批資料的長度。資料的特點是 長度不定,型別不定,由第乙個位元組 buffer 0 標識該資料的型別,共有2...

深入理解C C 函式指標

函式指標陣列的妙用 筆者在開發某軟體過程中遇到這樣乙個問題,前級模組傳給我二進位制資料,輸入引數為 char buffer和 int length,buffer是資料的首位址,length表示這批資料的長度。資料的特點是 長度不定,型別不定,由第乙個位元組 buffer 0 標識該資料的型別,共有2...

深入理解C C 函式指標

函式指標陣列的妙用 筆者在開發某軟體過程中遇到這樣乙個問題,前級模組傳給我二進位制資料,輸入引數為 char buffer和 int length,buffer是資料的首位址,length表示這批資料的長度。資料的特點是 長度不定,型別不定,由第乙個位元組 buffer 0 標識該資料的型別,共有2...