数组指针与指针数组的概念很容易混淆,数组指针本质上就是一个指针,它所指向的内容是一个数组,例如int (*p)[5]={0};
就是一个数组指针,其中p的类型为int (*)[5]
,而指针数组本质上是一个数组,如int * arr[5]={0};
是一个数组,数组里面每一个元素都是一个指针。
例如:
int a[10]={1,2,3,4};
int(*p)[10]=&a; //数组指针
函数指针,定义一个指针变量来存放函数地址,这个指针变量就叫作函数指针。
例如:
void test(){
printf("haha\n");
}
void(*pf)()=test; //函数指针
(*pf)(); //调用test函数
pf(); //调用test函数
int func(int a){
printf("hehe\n");
int(*pfunc)(int)=func;//函数指针
(*pfunc)(10); //调用func函数
pfunc(10); //调用func函数
在这里我们要注意函数指针与函数声明之间的区别,例如void(*pfunc1)();
为函数指针,而void *pfunc2 ();
为函数的声明。
在《C陷阱和缺陷》中有这样两行代码:
(*(void (*)())0)();
void (*signal(int , void(*)(int)))(int);
在(*(void (*)())0)();
中,void (*)()
代表函数指针,(void (*)())0
给0强制转换类型,0是一个地址,也就是说一个函数存在首地址为0的一块区域内,最外层的(* )()
代表了调用内层函数指针的函数,所以(*(void (*)())0)();
是一条函数调用语句,调用(void(*)())0
函数。
在void (*signal(int , void(*)(int)))(int);
中,void(*)(int)
是函数指针,和int一起作为signal函数的参数,signal函数的返回值是void(*)(int)
,将*signal(int , void(*)(int))
看作一个整体,再加上外层的void()(int)
就可以发现,其实void (*signal(int , void(*)(int)))(int);
是一条函数声明语句。
函数指针数组是一个数组,数组中存放的元素是一个个函数指针,例如int (*parr[10])();
parr是一个数组,数组的内容是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 divi(int a, int b) {
return a / b;
}
int main(){
int x;
int y;
int input = 1;
int ret = 0;
while (input){
printf("*************************\n");
printf(" 1:add 2:sub \n");
printf(" 3:mul 4:divi \n");
printf("*************************\n");
printf("请选择:\n");
scanf("%d", &input);
if (input < 1 || input > 4)
{
printf("输入错误,请重新输入!\n");
continue;
}
int(*p[5])(int, int) = { 0, add, sub, mul, divi }; //函数指针数组
printf("请输入操作数:\n");
scanf("%d %d", &x, &y);
ret = (*p[input])(x, y);
printf("ret = %d \n", ret);
}
system("pause");
return 0;
}