C语言中函数指针

函数调用的方法有两种分别如下:

void Fun(void)
{
    //do something
}

int main(void)
{
   ...
   Fun();
   ...
   return 0;
}
void Fun(void)
{
  //do something
}

int main(void)
{
   ...
   void (*p)(void);
   p=Fun;
   p();    
   return 0;
}

上面两种方式都可以调用函数,第二种方式在回调函数中用的最多,尤其是在操作系统的任务切换中。那么我们就重点来分析下第二种方式。

第二种方式申明了一个函数指针p,那为啥写成void(*p)(void)?为啥不直接写成*p,不是说了函数名就是该函数的入口地址,为啥不直接把该地址传给指针呢,这样既好理解又方便省事,其实这样想不无道理,但是在C语言中有变量类型这个东东,比如int赋值给char,编译器就会报错,同样的道理函数也有类型,那么既然函数有类型,我们如何知道变量类型呢。现在写一个程序如下,注意,程序有错误,这个错误是故意写的,目的就是让编译器报错,使用编译器为gcc。

#include "stdio.h"

void Fun(void)
{
    printf("hello world");    
}

int main(void)
{
    void *p;
        p=Fun;
    return 0;
}

下面是编译器输出:

g++.exe -x c++ -c C:\Users\Administrator.SC-201903191934\Desktop\unknown1.c -o C:\Users\Administrator.SC-201903191934\Desktop\unknown1.o -Wall -fpermissive -Wno-sign-compare -g
C:\Users\Administrator.SC-201903191934\Desktop\unknown1.c: In function `int main()':
C:\Users\Administrator.SC-201903191934\Desktop\unknown1.c:12: warning: invalid conversion from `void (*)()' to `void*'

编译器报错意思是不能讲void (*)()类型转化为void *类型。这个警告就说明了函数Fun的类型是(*)()类型,而申明的指针类型是*,两个类型不一致,故导致其错误,既然这样,那么如何申请一个指针类型为(*)()的呢?C语言中给了函数指针这个类型,即(*)()类型,因此为了让指针指向一个函数地址,只需要声明一个(*)()类型即可。

如下:

#include "stdio.h"

void Fun(void)
{
    printf("hello world"); 

}

int main(void)
{
void (*p)(void); //函数指针
p=Fun; return 0;
}

这样就将函数Fun的地址给指针p了,与其说p是一个函数指针,倒不如说p是一个函数名,只是没有什么其函数内部,这里可以从旧式的函数申明上可以看出。

因此要运行这个所谓的函数p,只需要和函数使用一样就行,p();

现在再来看一个代码:

void *Fun(void)
{
    printf("hello world");    
    return NULL;
}

这里函数Fun改为了有返回值的函数,其返回值为指针void *类型,那么我们如何来声明一个指针指该函数呢?

从上门面分析可以看出,该函数的类型应该是*(*)();所以现在我们需要声明一个相同类型的指针,即void *(*p)(void);这样就只需要p=Fun;就可以达到目的了。

#include "stdio.h"

void *Fun(void)
{
    printf("hello world");    
    return NULL;
}

int main(void)
{
    void *(*p)(void);
    p=Fun;
    return 0;
}

总结:1. 函数名也是有类型的,因此在C语言中,必须严格遵守类型匹配原则。

           2. 函数名的类型为(*)()类型,如果有返回值,则在最前面增加返回类型即可,例如

               int * *Fun(int,int );其函数类型为**(*)(int,int),对应的函数指针为int **(*p)(int,int);

posted @ 2019-06-08 17:48  LSWen  阅读(481)  评论(0编辑  收藏  举报