C语言——函数指针——函数指针数组 (详解)

函数指针数组 

函数指针数组的作用

函数指针数组是一个数组,其中的每个元素都是一个函数指针。函数指针是指向函数的指针变量,可以用来调用相应的函数。函数指针数组的作用是可以根据需要动态地选择并调用不同的函数。

函数指针数组的使用场景有很多,以下是一些常见的应用:

  1. 回调函数:函数指针数组可以用于实现回调机制,即在某个事件发生时,根据不同的情况选择并调用相应的回调函数。
  2. 状态机:函数指针数组可以用于实现状态机,即根据当前状态选择并调用相应的处理函数。
  3. 命令分发:函数指针数组可以用于实现命令分发,即根据接收到的命令选择并调用相应的处理函数。
  4. 策略模式:函数指针数组可以用于实现策略模式,即根据不同的策略选择并调用相应的处理函数。

总之,函数指针数组提供了一种灵活的方式来选择并调用不同的函数,可以根据具体需求动态地改变函数的行为。

——————————————————————————————————————————  

函数指针数组的语法格式

函数指针数组是一个数组,其中的元素都是函数指针。函数指针是指向函数的指针变量,可以用来调用相应的函数。函数指针数组的语法格式如下:

返回值类型 (*指针数组名[数组大小])(参数列表);

其中,返回值类型是函数的返回值类型,指针数组名是函数指针数组的名称,数组大小是指针数组中元素的个数,参数列表是函数的参数类型和个数。

举个例子,

如果我们有一个函数指针数组,其中包含两个元素,个元素都指向一个无返回值、无参数的函数,可以这样声明:

void (*funcPtrArray[2])();

这里,funcPtrArray是一个包含两个元素的函数指针数组,每个元素都是一个指向无返回值、无参数的函数的指针。

使用函数指针数组时,可以通过索引来访问数组中的元素,并通过函数指针调用相应的函数。

——————————————————————————————————————————  

函数指针数组语法格式的讲解

举例来说,我们可以定义一个函数指针数组,其中的元素指向不同的排序算法函数:

#include <stdio.h>

// 声明排序算法函数
void bubbleSort(int arr[], int size);
void selectionSort(int arr[], int size);
void insertionSort(int arr[], int size);

int main() {
    // 定义函数指针数组并初始化
    void (*sortFuncs[3])(int[], int) = {bubbleSort, selectionSort, insertionSort};
    //函数指针 函数指针定义三个 ,然后是(数组和,整形类型的)

    // 调用函数指针数组中的函数
    int arr[] = {5, 2, 8, 1, 9};
    int size = sizeof(arr) / sizeof(arr[0]);

    for (int i = 0; i < 3; i++) 
    {
        sortFuncs[i](arr, size);//这里主要是看这一段 这个代码是我随机从网上找的,主要是让你们看看这个语法格式的使用 ,下面会进行转移表,也就是计算器的使用版本2,就是采取函数指针数组的形势进行计算的
        printf("排序结果:");

        for (int j = 0; j < size; j++) 
        {
            printf("%d ", arr[j]);
        }

        printf("\n");
    }

    return 0;
}

// 冒泡排序
void bubbleSort(int arr[], int size) {
    // 排序逻辑...
}

// 选择排序
void selectionSort(int arr[], int size) {
    // 排序逻辑...
}

// 插入排序
void insertionSort(int arr[], int size) {
    // 排序逻辑...
}

——————————————————————————————————————————————————————————————————————————————————————

函数指针数组的实际使用(转移表(计算器))

这里会简单提一嘴,在指针4篇章还会进行详细的解释,指针4篇章会详细解释qsort函数的使用和构建模拟

计算器的模拟版本1

运算函数 

主函数 

计算器形成

但是之前的代码是有问题的,冗余性太强

#include<stdio.h>

// 函数原型声明
int add(int x, int y);
int subtract(int x, int y);
int multiply(int x, int y);
int divide(int x, int y);

int main() {
    int input;
    printf("请输入运算类型(1-加法,2-减法,3-乘法,4-除法,0-退出):");
    scanf("%d", &input);

    while (input != 0) {
        switch (input) {
            case 1:
                printf("请输入两个加数:");
                int x, y;
                scanf("%d %d", &x, &y);
                printf("%d\n", add(x, y));
                break;
            case 2:
                printf("请输入两个减数:");
                scanf("%d %d", &x, &y);
                printf("%d\n", subtract(x, y));
                break;
            case 3:
                printf("请输入两个乘数:");
                scanf("%d %d", &x, &y);
                printf("%d\n", multiply(x, y));
                break;
            case 4:
                printf("请输入两个数,其中第二个数为除数:");
                scanf("%d %d", &x, &y);
                if (y != 0) {
                    printf("%d\n", divide(x, y));
                } else {
                    printf("除数不能为0。\n");
                }
                break;
            default:
                printf("输入错误,请输入正确的运算类型。\n");
                break;
        }
        printf("是否继续?(0-否,其他值-是):");
        scanf("%d", &input);
    }

    printf("退出计算器。\n");
    return 0;
}

// 函数定义
int add(int x, int y) {
    return x + y;
}

int subtract(int x, int y) {
    return x - y;
}

int multiply(int x, int y) {
    return x * y;
}

int divide(int x, int y) {
    return x / y;
}

这里ai解释一下 不过多赘述 
这次计算机的目的是为了进行回调函数和函数指针数组的学习

这段C语言代码实现了一个简单的计算器,能够进行加法、减法、乘法和除法运算。下面是逐行解释:
1. `#include<stdio.h>`:这行代码包含了标准输入输出头文件`stdio.h`,它是C语言中用于输入输出函数的基础头文件。
2. 函数原型声明:这些是函数原型声明,指明了将要定义的四个函数`add`、`subtract`、`multiply`和`divide`的返回类型和参数类型。
3. `int main()`:这是主函数的定义,`main`是C程序的入口点。
4. `int input;`:这行代码定义了一个整数变量`input`,用于存储用户的选择。
5. `printf("请输入运算类型(1-加法,2-减法,3-乘法,4-除法,0-退出):");`:这行代码打印出提示信息,要求用户输入运算类型。
6. `scanf("%d", &input);`:这行代码使用`scanf`函数从用户那里读取一个整数,并存储在变量`input`中。
7. `while (input != 0) {`:这是一个`while`循环,它会在用户输入0之前一直执行。
8. `switch (input)`:这是一个`switch`语句,它根据变量`input`的值来执行不同的代码块。
9. `case 1:`:这是一个`case`标签,当`input`等于1时,执行后面的代码块。
10. `printf("请输入两个加数:");`:这行代码打印出提示信息,要求用户输入两个加数。
11. `scanf("%d %d", &x, &y);`:这行代码使用`scanf`函数从用户那里读取两个整数,并存储在变量`x`和`y`中。
12. `printf("%d\n", add(x, y));`:这行代码调用`add`函数,并打印出返回的结果。
13. `break;`:这个关键字用于退出`switch`语句。
14. `case 2:`:这是一个`case`标签,当`input`等于2时,执行后面的代码块。
15. `printf("请输入两个减数:");`:这行代码打印出提示信息,要求用户输入两个减数。
16. `scanf("%d %d", &x, &y);`:这行代码使用`scanf`函数从用户那里读取两个整数,并存储在变量`x`和`y`中。
17. `printf("%d\n", subtract(x, y));`:这行代码调用`subtract`函数,并打印出返回的结果。
18. `break;`:这个关键字用于退出`switch`语句。
19. `case 3:`:这是一个`case`标签,当`input`等于3时,执行后面的代码块。
20. `printf("请输入两个乘数:");`:这行代码打印出提示信息,要求用户输入两个乘数。
21. `scanf("%d %d", &x, &y);`:这行代码使用`scanf`函数从用户那里读取两个整数,并存储在变量`x`和`y`中。
22. `printf("%d\n", multiply(x, y));`:这行代码调用`multiply`函数,并打印出返回的结果。
23. `break;`:这个关键字用于退出`switch`语句。
24. `case 4:`:这是一个`case`标签,当`input`等于4时,执行后面的代码块。
25. `printf("请输入两个数,其中第二个数为除数:");`:这行代码打印出提示信息,要求用户输入两个数,其中第二个数为除数。
26. `scanf("%d %d", &x, &y);`:这行代码使用`scanf`函数从用户那里读取两个整数,并存储在变量`x`和`y`中。
27. `if (y != 0) {`:这是一个`if`语句,它检查除数`y`是否不为0。
28. `printf("%d\n", divide(x, y));`:这行代码调用`

———————————————————————————————————————————

计算器的模拟版本2(函数指针数组的使用)(回调函数)

简化

冗余

老的代码的问题就是

冗余

写死

不能完成不同的任务

函数调用的时候只需要知道地址就可以

calc计算器

这里也称之为转移表

#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
int add(int x, int y)
{
	return x + y;
}
int subtract(int x, int y)
{
	return x - y;
}
int multiply(int x, int y)
{
	return x * y;
}
int except(int x, int y)
{
	return x / y;
}
void meun()
{
	printf("**************************\n");
	printf("*****    0  退出    ******\n");
	printf("*****    1  加法    ******\n");
	printf("*****    2  减法    ******\n");
	printf("*****    3  乘法    ******\n");
	printf("*****    4  除法    ******\n");
	printf("**************************\n");
}
void cala(int* compute(int, int))
{
	int x = 0; int y = 0;
	int pf = 0;
	scanf("%d %d", &x, &y);
	pf = compute(x, y);
	printf("%d ", pf);
}
int main()
{
	meun();
	int input = 0;
	scanf("%d", &input);	
	switch (input)
	{
	case 0:
		break;
	case 1:
		cala(add);
		break;
	case 2:
		cala(subtract);
		break;
	case 3:
		cala(subtract);
		break;
	case 4:
		cala(except);
		break;
	default:
		break;
	}
}

这里 用ai 逐行解释一下 我不过多作图解释了:
1. `#define _CRT_SECURE_NO_WARNINGS 1`:这是一个预处理器指令,用于告诉编译器忽略与安全相关的警告
2. `#include<stdio.h>`:这是一个预处理器指令,用于包含标准输入输出头文件`stdio.h`,这是C语言中用于输入输出函数的基础头文件。
3. `int add(int x, int y)`:这是一个函数定义,`add`是一个函数名,它接受两个整数参数`x`和`y`,并返回它们的和。
4. `int subtract(int x, int y)`:这是另一个函数定义,`subtract`是一个函数名,它接受两个整数参数`x`和`y`,并返回它们的差。
5. `int multiply(int x, int y)`:这是另一个函数定义,`multiply`是一个函数名,它接受两个整数参数`x`和`y`,并返回它们的积。
6. `int except(int x, int y)`:这是另一个函数定义,`except`是一个函数名,它接受两个整数参数`x`和`y`,并返回它们的商。注意,这个函数没有进行除数是否为0的检查,这在实际使用中是不安全的。
7. `void meun()`:这是一个函数定义,`meun`是一个函数名,它没有返回值。这个函数的作用是打印出菜单供用户选择运算类型。
8. `void cala(int* compute(int, int))`:这是另一个函数定义,`cala`是一个函数名,它接受一个指向函数的指针作为参数。这个函数的作用是调用用户提供的运算函数,并打印出结果。
9. `int main()`:这是主函数的定义,`main`是一个特殊的关键字,表示程序的入口点。
10. `meun();`:这行代码调用`meun`函数,打印出菜单。
11. `int input = 0;`:这行代码定义了一个整数变量`input`并初始化为0,用于存储用户的选择。
12. `scanf("%d", &input);`:这行代码使用`scanf`函数从用户那里读取一个整数,并存储在变量`input`中。
13. `switch (input)`:这是一个`switch`语句,它根据变量`input`的值来执行不同的代码块。
14. `case 0:`:这是一个`case`标签,当`input`等于0时,执行后面的代码块。
15. `break;`:这个关键字用于退出`switch`语句。
16. `case 1:`:这是一个`case`标签,当`input`等于1时,执行后面的代码块。
17. `cala(add);`:这行代码调用`cala`函数,并传递`add`函数的地址作为参数。
18. `break;`:这个关键字用于退出`switch`语句。
19. `case 2:`:这是一个`case`标签,当`input`等于2时,执行后面的代码块。
20. `cala(subtract);`:这行代码调用`cala`函数,并传递`subtract`函数的地址作为参数。
21. `break;`:这个关键字用于退出`switch`语句。
22. `case 3:`:这是一个`case`标签,当`input`等于3时,执行后面的代码块。
23. `cala(subtract);`:这行代码调用`cala`函数,并传递`subtract`函数的地址作为参数。这里似乎是一个错误,因为`subtract`函数并不适合乘法操作。
24. `break;`:这个关键字用于退出`switch`语句。
25. `case 4:`:这是一个`case`标签,当`input`等于4时,执行后面的代码块。
26. `cala(except);`:这行代码调用`cala`函数,并传递`except`函数的地址作为参数。
27. `break;`:

—————————————————————————————————————————— 

版本3(函数指针数组的使用)

计算机代码的简化

创建函数指针的数组

#define _CRT_SECURE_NO_WARNINGS 1
#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
int add(int x, int y)
{
	return x + y;
}
int subtract(int x, int y)
{
	return x - y;
}
int multiply(int x, int y)
{
	return x * y;
}
int except(int x, int y)
{
	return x / y;
}
void meun()
{
	printf("**************************\n");
	printf("*****    0  退出    ******\n");
	printf("*****    1  加法    ******\n");
	printf("*****    2  减法    ******\n");
	printf("*****    3  乘法    ******\n");
	printf("*****    4  除法    ******\n");
	printf("**************************\n");
}
int main()
{

	int (*compute[5])(int, int) = { 0, add,subtract,multiply,except };
	int input = 0; 

	do
	{
		meun();
		printf("请选择:\n");

		scanf("%d", &input);
		printf("请输入:");
		if (input>=1&& input<=4)
		{
			int x = 0; int y = 0; int zd = 0;
			scanf("%d %d", &x, &y);
			zd = (*compute[input])(x, y);
			printf("%d\n", zd);
		}
		else if (input == 0)
		{
			printf("退出程序。");
			break;
		}
		else 
		{
			printf("输入错误。");
		}

	} while (input);
}

这里 用ai 逐行解释一下 我不过多作图解释了:
这段代码是C语言编写的一个简单计算器程序,下面是逐行解释:
1. `#define _CRT_SECURE_NO_WARNINGS 1`:这是一个预处理器指令,用于告诉编译器忽略与安全相关的警告,特别是在使用像`fgets`、`fopen`等函数时。这行代码应该只包含一次,重复定义可能会导致编译错误。
2. `#include<stdio.h>`:这是一个预处理器指令,用于包含标准输入输出头文件`stdio.h`,这是C语言中用于输入输出函数的基础头文件。
3. `int add(int x, int y)`:这是一个函数定义,`add`是一个函数名,它接受两个整数参数`x`和`y`,并返回它们的和。
4. `int subtract(int x, int y)`:这是另一个函数定义,`subtract`是一个函数名,它接受两个整数参数`x`和`y`,并返回它们的差。
5. `int multiply(int x, int y)`:这是另一个函数定义,`multiply`是一个函数名,它接受两个整数参数`x`和`y`,并返回它们的积。
6. `int except(int x, int y)`:这是另一个函数定义,`except`是一个函数名,它接受两个整数参数`x`和`y`,并返回它们的商。注意,这个函数没有进行除数是否为0的检查,这在实际使用中是不安全的。
7. `void meun()`:这是一个函数定义,`meun`是一个函数名,它没有返回值。这个函数的作用是打印出菜单供用户选择运算类型。
8. `int main()`:这是主函数的定义,`main`是一个特殊的关键字,表示程序的入口点。
9. `int (*compute[5])(int, int) = { 0, add,subtract,multiply,except }`:这行代码定义了一个函数指针数组`compute`,长度为5。每个元素指向一个算术运算函数。数组的第一个元素被初始化为`0`,这通常用于指针数组的终止标志。
10. `int input = 0;`:这行代码定义了一个整数变量`input`并初始化为0,用于存储用户的选择。
11. `do`:这是一个循环开始关键字,表示开始一个`do-while`循环。
12. `meun();`:这行代码调用`meun`函数,打印出菜单。
13. `printf("请选择:\n");`:这行代码打印出提示信息“请选择:”。
14. `scanf("%d", &input);`:这行代码使用`scanf`函数从用户那里读取一个整数,并存储在变量`input`中。
15. `printf("请输入:");`:这行代码打印出提示信息“请输入:”。
16. `if (input>=1&& input<=4)`:这是一个条件判断,检查用户输入是否在1到4之间,包括1和4。
17. `{ int x = 0; int y = 0; int zd = 0;`:这个大括号开始了一个代码块,用于执行当用户输入在1到4之间时的操作。
18. `scanf("%d %d", &x, &y);`:这行代码使用`scanf`函数从用户那里读取两个整数,并存储在变量`x`和`y`中。
19. `zd = (*compute[input])(x, y);`:这行代码调用相应的运算函数,并将结果存储在变量`zd`中。
20. `printf("%d\n", zd);`:这行代码打印出计算结果。
21. `}`:这个大括号结束了上面的代码块。
22. `else if (input == 0)`:这是另一个条件判断,检查用户输入是否为0。
23. `{ printf("退出程序。");`:这个代码块用于执行当用户输入为0时的操作,打印出提示信息。
24. `break;`:这个关键字用于退出循环。
25. `}`:这个大括号结束了上面的条件判断。
26. `else`:这是另一个条件判断,用于处理用户输入不在1到4之间

————————————————————————————————————————————————————————————————————————————————————————————————————————————————————————————————————————————————————————————————————————————

计算器代码总结 

#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
//菜单
void meun()
{
	printf("**************************\n");
	printf("*****    0  退出    ******\n");
	printf("*****    1  加法    ******\n");
	printf("*****    2  减法    ******\n");
	printf("*****    3  乘法    ******\n");
	printf("*****    4  除法    ******\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 dive(int x, int y) 
{
    return x / y;
}

//函数指针数组的使用
void Function_pointers()
{
	int(*compute[5])(int, int) = { 0,add,sub,mul,dive };
	int input = 0; int x = 0; int y = 0;
	do
	{
		if (input > 0 && input <= 4)
		{
			meun();
			printf("请输入>\n");
			scanf("%d", &input);
			printf("请输入计算数值>\n");
			scanf("%d %d", &x, &y);
			int outcome = (*compute[input])(x, y);
			printf("结果是:%d\n", outcome);
		}
		else if (input == 0)
		{
			printf("退出");
			break;
		}
		else
		{
			printf("输入错误。");
		}

	} while (input);
}

//回调函数的使用(依旧是指针函数数组)
void invoke(int*ps(int, int))
{
	int x = 0; int y = 0;
	printf("请输入计算数值>\n");
	scanf("%d %d", &x, &y);
	int outcome = (*ps)(x, y);
	printf("结果是:%d\n", outcome);

}
void Callback()
{
	int input = 0;
	do
	{
		meun();
		printf("请输入>\n");
		scanf("%d", &input);

		switch (input)
		{
		case 0:
			break;
		case 1:
			invoke(add);
			break;
		case 2:
			invoke(sub);
			break;
		case 3:
			invoke(mul);
			break;
		case 4:
			invoke(dive);
			break;
		default:
			printf("输入错误。");
			break;
		}
	} while (input);
}
//正常计算函数
void count()
{
	int input = 0;
	do
	{
		int x = 0; int y = 0;
		meun();
		printf("请输入>\n");
		scanf("%d", &input);
		int outcomt = 0;
		switch (input)
		{
		case 0:
			printf("退出");
			break;
		case 1:
			printf("请输入计算数值>\n");
			scanf("%d %d", &x, &y);
			outcomt = add(x, y);
			printf("结果是:%d\n", outcomt);
			break;
		case 2:
			printf("请输入计算数值>\n");
			scanf("%d %d", &x, &y);
			outcomt = sub(x, y);
			printf("结果是:%d\n", outcomt);
			break;
		case 3:
			printf("请输入计算数值>\n");
			scanf("%d %d", &x, &y);
			outcomt = mul(x, y);
			printf("结果是:%d\n", outcomt);
			break;
		case 4:
			printf("请输入计算数值>\n");
			scanf("%d %d", &x, &y);
			outcomt = dive(x, y);
			printf("结果是:%d\n", outcomt);
			break;
		default:
			printf("输入错误。");
			break;
		}
	} while (input);
}



//主函数
int main()
{
	//printf("函数指针数组的使用>\n");
	//Function_pointers();

	//printf("回调函数的使用>\n");
	//Callback();

	printf("正常函数的使用>\n");
	count();
}

 

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值