一、 函数指针数组
我们都知道 数组是一个存放相同类型数据的存储空间 那我们已经学习了指针数组
那么函数有没有对应的指针数组呢? 如果有那应该怎么定义呢?
1. 函数指针数组的定义
我们说 函数指针数组的定义 应该遵循以下格式
int (*p[10])();
首先p和[]结合说明了它是一个数组
我们再将它拿掉可以发现 数组的内容就是
int (*)() 类型的函数指针。
2. 函数指针数组的用途:转移表(以计算器为例)
我们首先来写一段代码实现计算器的功能
void mune()
{printf("****** 1.add 2.sub *****\n");printf("****** 3.mul 4.div *****\n");printf("****** 0.exit *****\n");printf("**************************\n");
}
int Add(int x, int y)
{return x + y;
}
int Sub(int x, int y)
{return x - y;
}
int Mul(int x, int y)
{return x * y;
}
int Div(int x, int y)
{return x / y;
}int main()
{int input = 0;int x = 0;int y = 0;int ret = 0;do{mune();printf("请选择:>");scanf("%d",&input);switch (input){case 1:printf("请输入两个整数\n");scanf("%d %d", &x, &y);ret = Add(x, y);printf("%d\n", ret);break;case 2:printf("请输入两个整数\n");scanf("%d %d", &x, &y);ret = Sub(x, y);printf("%d\n", ret);break;case 3:printf("请输入两个整数\n");scanf("%d %d", &x, &y);ret = Mul(x, y);printf("%d\n", ret);break;case 4:printf("请输入两个整数\n");scanf("%d %d", &x, &y);ret = Div(x, y);printf("%d\n", ret);break;case 0:printf("退出计算器\n");break;default:printf("选择错误,请重新选择\n");}} while (input);return 0;
}
那么我们再使用函数指针来实现试试
void mune()
{printf("****** 1.add 2.sub *****\n");printf("****** 3.mul 4.div *****\n");printf("****** 0.exit *****\n");printf("**************************\n");
}
int Add(int x, int y)
{return x + y;
}
int Sub(int x, int y)
{return x - y;
}
int Mul(int x, int y)
{return x * y;
}
int Div(int x, int y)
{return x / y;
}
//函数指针数组存放上述函数
//转移表
int (*pf[5])(int x, int y) = { NULL, Add, Sub, Mul, Div };int main()
{int input = 0;int x = 0;int y = 0;int ret = 0;do{mune();printf("请选择:>");scanf("%d", &input);if (input == 0){printf("退出计算器\n");}else if (input >= 1 && input <= 4){printf("请输入两个操作符:>");scanf("%d %d", &x, &y);ret = pf[input](x, y);printf("%d\n", ret);}else{printf("选择错误\n");}} while (input);return 0;
}
二、回调函数
我们还是以计算器为例来解释回调函数的使用
void mune()
{printf("****** 1.add 2.sub *****\n");printf("****** 3.mul 4.div *****\n");printf("****** 0.exit *****\n");printf("**************************\n");
}
int Add(int x, int y)
{return x + y;
}
int Sub(int x, int y)
{return x - y;
}
int Mul(int x, int y)
{return x * y;
}
int Div(int x, int y)
{return x / y;
}int main()
{int input = 0;int x = 0;int y = 0;int ret = 0;do{mune();printf("请选择:>");scanf("%d",&input);switch (input){case 1:printf("请输入两个整数\n");scanf("%d %d", &x, &y);ret = Add(x, y);printf("%d\n", ret);break;case 2:printf("请输入两个整数\n");scanf("%d %d", &x, &y);ret = Sub(x, y);printf("%d\n", ret);break;case 3:printf("请输入两个整数\n");scanf("%d %d", &x, &y);ret = Mul(x, y);printf("%d\n", ret);break;case 4:printf("请输入两个整数\n");scanf("%d %d", &x, &y);ret = Div(x, y);printf("%d\n", ret);break;case 0:printf("退出计算器\n");break;default:printf("选择错误,请重新选择\n");}} while (input);return 0;
}
我们可以发现 case1 2 3 4里面有大量冗余的代码
那么我们可不可以通过一定的手段解决这个问题呢?
答案是可以 就是使用回调函数
定义
回调函数就是一个被作为参数传递的函数。
我们来设计一个case_()函数 并且将add div这些函数作为参数传递进去
要求: 可以实现和原函数一样的功能 代码不能冗余
void calc(int (*pf)(int x, int y))
{int x = 0;int y = 0;int ret = 0;printf("请输入两个操作符:>\n");scanf("%d %d", &x, &y);ret = pf(x, y);printf("%d\n", ret);
}
这就是上面思想指导写出来的代码
完整代码如下
void mune()
{printf("****** 1.add 2.sub *****\n");printf("****** 3.mul 4.div *****\n");printf("****** 0.exit *****\n");printf("**************************\n");
}
int Add(int x, int y)
{return x + y;
}
int Sub(int x, int y)
{return x - y;
}
int Mul(int x, int y)
{return x * y;
}
int Div(int x, int y)
{return x / y;
}
void calc(int (*pf)(int x, int y))
{int x = 0;int y = 0;int ret = 0;printf("请输入两个操作符:>\n");scanf("%d %d", &x, &y);ret = pf(x, y);printf("%d\n", ret);
}int main()
{int input = 0;do{mune();printf("请选择:>");scanf("%d",&input);switch (input){case 1:calc(Add);break;case 2:calc(Sub);break;case 3:calc(Mul);break;case 4:calc(Div);break;case 0:printf("退出计算器\n");break;default:printf("选择错误,请重新选择\n");}} while (input);return 0;
}
让我们试试它的运行结果
可以完美运行的