一、函数指针
顾名思义,函数指针,首先它是一个指针,因为可以指向函数,所以称为函数指针。可以与数组指针类比。
与定义数组指针类似,函数指针的定义如下:
函数返回值类型(*变量名)(函数参数类型)
如 void (*pfun)(int) //函数指针pfun指向的函数的类型是 void (int)
来判断下面的代码是什么意思
(1)、char * (*fun1)(char * p1, char * p2);
(2)、char * *fun2(char * p1, char * p2);
(3)、char * fun3(char * p1, char * p2);
上述代码只要考虑优先级,我们就很容易判断出来。
解:
(1)、*先与fun1结合,所以fun1是一个指针,去掉(*fun1),就是fun1可以指向的类型,所以fun1是一个函数指针。可以指向的函数的类型为char*(char* p1,char* p2)。故第(1)行语句是定义一个函数指针。
(2)、()的优先级高于*故fun2先与()结合,所以fun2是一个函数,函数的返回值类型是char**,参数数是(char*p1,char*p2),所以这是一个函数的声明。
(3)、同(2)一样fun3是一个函数,这一一个函数的声明。
那么,函数指针有什么用呢?函数指针与普通指针一样解引用就可一访问到指向的对象。解引用函数指针就可以调用函数。
#include <stdio.h> int add(int a,int b) { return a+b; } int main() { int(*padd)(int,int)=&add;//定义函数指针padd指向add() int ret=0; //ret=add(2,3); ret=(*padd)(2,3); printf("%d\n",ret); return 0; }
二、函数指针数组
顾名思义,函数指针数组,首先是一个数组,因为数组元素是函数指针,所以称为函数指针数组。
1、函数指针数组的定义
该如何定义一个函数指针数组呢?
我们先来看一下普通数组和指针数组是如何定义的。
char arr1[3]; //字符数组
char *arr2[3]; //指针数组
同理,char (*arr[3]) (char*); //函数指针数组
其实很好理解,[]优先级高于*,arr先与[3]结合,所以arr是一个数组,大小为3;去掉arr[3],就是数组元素的类型,char(*)(char*)这是一个函数指针类型。故arr是一个函数指针数组大小为3.
2、数组的初始化
与初始化普通数组一样,只是数组的元素是函数的地址
比如数组元素是指向int (int,int)类函数
int(*pfun[3])(int,int)={fun1,fun2,fun3}; 也可以{&fun1,&fun2,&fun3}
3、函数指针数组的应用(简易计算器的实现)
#include <stdio.h> 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 a/b; } int main() { int input=1; int x=0,y=0; int ret=0; int(*p[4])(int,int)={add,sub,mul,div}; while(input) { printf("1.add 2.sub 3.mul 4.div 0.exit\n"); scanf("%d",&input); if(input>0&&input<5) { printf("请输入操作数\n"); scanf("%d%d",&x,&y); ret=(*p[intput-1])(x,y); printf("%d\n",ret); } else if(input==0) break; else printf("输入有误,请重新输入\n"); } return 0; }
上面的代码是使用函数指针数组来实现计算器,不用函数指针数组还可使用switch语句,但是如果计算器的功能复杂就会使用大量的case语句,这种情况下使用函数指针数组就可以减少代码冗余。
三、函数指针数组的数组指针
顾名思义,函数指针数组的数组指针,首先它是一个指针,因为它指向一个数组,所以叫数组指针,又这个数组是一个函数指针数组,所以这个指针就是指向函数指针数组的数组指针。
如下是定义一个简单的函数指针数组的数组指针:
char * (*(*parr)[3])(char * p);
*先与parr结合,所以parr是一个指针,去掉(*parr)就是parr可以指向的类型,
char*(*[3])(char*p): *先与[3]结合,这是一个指针数组,parr指向一个数组,去掉[3] 就是数组的类型,char*(*)(char*p)这是一个函数指针类型,故parr是指向函数指针数组的数组指针。
只需对一个函数指针数组取地址,取出的地址就可以对函数指针数组的指针初始化或赋值。
四、回调函数
1、概念
什么是回调函数呢?
回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用为调用它所指向的函数时,我们就说这是回调函数。
既然通过函数指针就可以调用一个函数,那么我们是不是可以封装一个函数,既可以实现加减,又可以实现乘除呢?答案是可以的。(这里说的不适在函数内部判断来实现的)
//回调函数 #include <stdio.h> 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 a/b; } int op(int (*pfun)(int , int)) //回调函数op,只要给该函数传上面函数的地址, //就可以实现两个数的加减乘除 { return pfun(1,2); } int main() { int input=1; int x=0,y=0; int ret=0; int(*p[4])(int,int)={add,sub,mul,div}; while(input) { printf("1.add 2.sub 3.mul 4.div 0.exit\n"); scanf("%d",&input); if(input>0&&input<5) { printf("请输入操作数\n"); scanf("%d%d",&x,&y); ret= op(p[input-1]); printf("%d\n",ret); } else if(input==0) break; else printf("输入有误,请重新输入\n"); } return 0; }
其实上面的例子没有必要用回调函数来实现,这样太复杂来,而且太大材小用了,只是方面理解。
2、应用
接下来我们来看一下回调函数的应用
编写一个冒泡排序函数可以既可以排序字符串,又可以排序数字,还可以排序字符。
#include <stdio.h> #include <string.h> int com_int(const void *elem1, const void *elem2 ) { return *(int *)elem1 - *(int *)elem2; } int com_str(const void *elem1, const void *elem2) { return strcmp((char *)*(int *)elem1, (char *)*(int *)elem2); } int com_char(const void *elem1, const void *elem2) { return*(char*)elem1 - *(char*)elem2; } void Swap(char *p1, char *p2, size_t sz) { size_t i = 0; for(i = 0; i<sz; i++) { char tmp = *(p1+i); *(p1+i) = *(p2+i); *(p2+i) = tmp; } } void bubble_sort(void *base, size_t num, size_t width,int (*cmp)(const void *elem1, const void *elem2)) { size_t i = 0,j=0; for(i = 0; i < num-1; i++) { for(j = 0; j < num-1-i; j++) { if(cmp((char *)base+ width*j, (char *)base+width*(j+1)) > 0) { Swap((char *)base+ width*j, (char *)base+width*(j+1), width); } } } } int main() { //char *arr1[] = {"bbbb","aaaa","dddd","cccc"}; //int arr1[] = { 1, 3, 5, 7, 2, 4, 6, 8 }; char arr1[] = "asdggt"; int i = 0,len=0; len = sizeof(arr1) / sizeof(arr1[0]); bubble_sort(arr1,len , sizeof(char), com_char); for(i = 0 ;i<sizeof(arr1)/sizeof(arr1[0]);i++) { printf("%c\n",arr1[i]); } printf("\n"); return 0; }
本文出自 “牛丽” 博客,请务必保留此出处http://15129279495.blog.51cto.com/10845420/1737892
原文地址:http://15129279495.blog.51cto.com/10845420/1737892