一、 函数指针数组
我们都知道 数组是一个存放相同类型数据的存储空间 那我们已经学习了指针数组
那么函数有没有对应的指针数组呢? 如果有那应该怎么定义呢?
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; }
让我们试试它的运行结果
可以完美运行的