天天看點

函數指針的使用和void *類型

1、C語言裡有void指針,可以指向任何類型,在使用前需要強制轉化類型。

#include <bits/stdc++.h>
using namespace std;
const int maxn = 500 + 10;
void fun(void *a)
{
    printf("%.2lf\n",*(double *)a);
}
int main(){
    double a=1.21;
    fun(&a);
    return 0;
}      

輸出就是1.21

2、void *fun()

這是一個傳回值為void指針的函數。

3、函數指針的應用,形式1:傳回類型(*函數名)(參數表) 

char (*pFun)(int); 
char glFun(int a){ return;} 
void main() 
{ 
    pFun = glFun; 
    (*pFun)(2); 
}      

  第一行定義了一個指針變量pFun。首先我們根據前面提到的“形式1”認識到它是一個指向某種函數的指針,這種函數參數是一個int型,傳回值是char類型。隻有第一句我們還無法使用這個指針,因為我們還未對它進行指派。

  第二行定義了一個函數glFun()。該函數正好是一個以int為參數傳回char的函數。我們要從指針的層次上了解函數——函數的函數名實際上就是一個指針,函數名指向該函數的代碼在記憶體中的首位址

   然後就是main()函數了,它的第一句您應該看得懂了——它将函數glFun的位址指派給變量pFun。main()函數的第二句中“*pFun”顯然是取pFun所指向位址的内容,當然也就是取出了函數glFun()的内容,然後給定參數為2。

4、使用typedef簡化:形式1:typedef  傳回類型(*新類型)(參數表)

typedef char (*PTRFUN)(int); 
PTRFUN pFun; 
char glFun(int a){ return;} 
void main() 
{ 
    pFun = glFun; 
    (*pFun)(2); 
}      

typedef的功能是定義新的類型。第一句就是定義了一種PTRFUN的類型,并定義這種類型為指向某種函數的指針,這種函數以一個int為參數并傳回char類型。後面就可以像使用int,char一樣使用PTRFUN了。

第二行的代碼便使用這個新類型定義了變量pFun,此時就可以像使用形式1一樣使用這個變量了。

來個練習:typedef void (*fun)( void ) ;是什麼?

答:

void (*Fun)(void);

假如就隻是寫上這樣的一句代碼,顯然這就是定義了一個變量Fun。

這個Fun變量是一個指針,指向 傳回值和參數都是空的函數的指針。

typedef void (*Fun)(void);

這樣的話Fun代表的就不是變量了,而是代表了一種新的類型。

用Fun來定義一個變量 如

Fun p;

上面這句代碼就定義了一個 指針 這個p指針和 最上面的那個Fun是同一個類型的變量。(p是指向 傳回值和參數都是空的函數的指針)。

5、最後一個大栗子來當個執行個體

#include <stdio.h>
#include <assert.h>
 
typedef int (*FP_CALC)(int,int);//定義一個函數指針類型
 
int add(int a, int b)
{
    return a + b;
}
 
int sub(int a, int b)
{
    return a - b;
}
 
int mul(int a, int b)
{
    return a * b;
}
 
int div(int a, int b)
{
    return b ? a/b : -1;
}
 
//定義一個函數,參數為op,傳回一個指針,該指針類型為擁有兩個int參數、
//傳回類型為int的函數指針。它的作用是根據操作符傳回相應函數的位址
FP_CALC calc_func(char op)
{
    switch( op )
    {
    case '+':
        return add;
    case '-':
        return sub;
    case '*':
        return mul;
    case '/':
        return div;
    default:
        return NULL;
    }
    return NULL;
}
 
//s_calc_func為函數,它的參數是 op,   
//傳回值為一個擁有兩個int參數、傳回類型為int的函數指針  
int (*s_calc_func(char op)) (int , int)
{
    return calc_func(op);
}
 
//最終使用者直接調用的函數,該函數接收兩個int整數,
//和一個算術運算符,傳回兩數的運算結果
int calc(int a, int b, char op)
{
    FP_CALC fp = calc_func(op);
    int (*s_fp)(int,int) = s_calc_func(op);//用于測試
 
    assert(fp == s_fp);// 可以斷言這兩個是相等的
 
    if(fp)
        return fp(a,b);
    else
        return -1;
}
 
void main()
{
    int a = 100, b = 20;
 
    printf("calc(%d, %d, %c) = %d\n", a, b, '+', calc(a, b, '+'));
    printf("calc(%d, %d, %c) = %d\n", a, b, '-', calc(a, b, '-'));   
    printf("calc(%d, %d, %c) = %d\n", a, b, '*', calc(a, b, '*'));   
    printf("calc(%d, %d, %c) = %d\n", a, b, '/', calc(a, b, '/')); 
}      

上面代碼中:

#if 0   
int (*s_calc_func(char op)) (int , int)  
{  
      return calc_func(op);  
}  
#endif
//改寫為下面形式,就很好了解
FP_CALC s_calc_func(char op)
{
    return calc_func(op);
}