前言
🎈个人主页:🎈 :✨✨✨初阶牛✨✨✨
🐻推荐专栏: 🍔🍟🌯 c语言进阶
🔑个人信条: 🌵知行合一
🍉本篇简介:>:介绍c语言中有关指针更深层的知识.
金句分享:
✨在该奋斗的岁月里,✨
✨对得起每一寸光阴.✨
目录
- 前言
- 一、函数指针
- 1.1 函数指针的创建
- 1.2 函数指针应用
- 二、函数指针数组
- 三、函数指针数组指针
- 四、回调函数
一、函数指针
1.1 函数指针的创建
函数指针,顾名思义,就一个指向函数的指针.
那么
整形指针是接收整形的地址;
字符指针是接收字符的地址.
数组指针是接收数组的地址
函数有地址吗?函数名又可以代表什么?
示例:自定义一个加法函数,观察函数的地址
//函数指针
#include <stdio.h>
int add(int a,int b)//自定义一个加法函数
{
return a + b;
}
int main()
{
int a = 2, b = 3;
int c=add(a, b);
printf("%d\n", c);//确认一下函数是否正确
printf("%p\n", &add);//取出函数的地址
printf("%p\n", add);//用数组名打印
return 0;
}
运行结果:(在不同环境下地址会不一样,这里采用的是vs x86环境)
5
00E61023
00E61023
不难发现,函数名也可以代表函数的地址.
那么函数指针该怎么写呢?
以数组指针为例:
数组指针示例:写一个指向 int arr[10] 数组的数组指针;
第一步:
(*p) //先确定是一个指针
第二步:
(*p)[10] //确定指向的是一个有10个元素的数组
第三步:
int(*p)[10] //确定该数组元素为int型
第四步:
int(*p)[10]=&arr;//将数组的地址赋值给数组指针
//或者int(*p)[10]=arr;
示例2:
函数指针:指向int add(int a,int b)
第一步:
(*p) //先确定是一个指针
第二步:
(*p)(int,int) //确定指向的函数有两个参数
第三步:
int (*p)(int,int) //确定该函数的返回类型
第四步:
int (*p)(int,int)=&add;//将函数的地址赋值给函数指针
//等价于:int (*p)(int,int)=add;
1.2 函数指针应用
那么函数指针有什么用吗?
当然是用来调用函数了.
同样以数组指针为例:
🌰使用数组指针访问数组.
int main()
{
int arr[10] = { 1,2,3,4,5 };
int(*p)[10] = arr;
for (int i = 0; i < 5; i++)
{
printf("%d ", ( * p)[i]);
}
printf("\n");
return 0;
}
🌰使用函数指针调用函数
#include <stdio.h>
int add(int a,int b)
{
return a + b;
}
int main()
{
int a = 2, b = 3;
int(*p1)(int, int) = &add;
int(*p2)(int, int) = add;
int ret1=add(a,b);
int ret2=(*p1)(a, b);//写法一
int ret3=p1(a, b);//写法2
printf("ret1=%d\nret2=%d\nret3=%d",ret1,ret2,ret3);
return 0;
}
运行结果:都能完成函数的调用
ret1=5
ret2=5
ret3=5
二、函数指针数组
函数指针数组实质用于存放函数指针的数组.
即数组的成员都是函数指针.
写法:以数组成员都是指向类似int add(int,int)的函数指针为例
1.先写出函数指针
(int) (*p)(int,int)
2.改成数组
(int) (*p[10])(int,int)//函数指针数组
题目:使用c语言制作简单的计算器.
要求,可以进行加减乘除运算.
//简易计算器
#include <stdio.h>
void menu()//菜单
{
printf("作者:初阶牛\n");
printf(" 欢迎使用计算器:\n");
printf("+-------------------------------+\n");
printf("| 1.add 2.sub |\n");
printf("| 3.mul 4.div |\n");
printf("| 0.退出 |\n");
printf("+-------------------------------+\n");
}
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 x=0, y=0;
int input = 1;
int ret = 0;
do
{
menu();//打印菜单
scanf("%d", &input);
switch (input)
{
case 1:
printf("输入两个操作数:");
scanf("%d %d", &x, &y);
ret = add(x, y);
printf("ret = %d\n", ret);
break;
case 2:
printf("输入两个操作数:");
scanf("%d %d", &x, &y);
ret = sub(x, y);
printf("ret = %d\n", ret);
break;
case 3:
printf("输入两个操作数:");
scanf("%d %d", &x, &y);
ret = mul(x, y);
printf("ret = %d\n", ret);
break;
case 4:
printf("输入两个操作数:");
scanf("%d %d", &x, &y);
ret = div(x, y);
printf("ret = %d\n", ret);
break;
case 0:
printf("退出计算器\n");
break;
default:
printf("很遗憾,并没有这个选项:\n");
break;
}
} while (input);
return 0;
}
对于上面的代码,我们可以使用函数指针来简化.
感受一下函数的指针的魅力吧!
#include <stdio.h>
void menu()
{
printf("作者:初阶牛\n");
printf(" 欢迎使用计算器:\n");
printf("+-------------------------------+\n");
printf("| 1.add 2.sub |\n");
printf("| 3.mul 4.div |\n");
printf("| 0.退出 |\n");
printf("+-------------------------------+\n");
}
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 x = 0, y = 0;
int input = 1;
int ret = 0;
int(*arr[5])(int, int)={0,add,sub,mul,div};//使用函数指针数组
do
{
menu();
scanf("%d", &input);
if (input > 0 && input < 5)
{
printf("输入两个操作数:");
scanf("%d %d", &x, &y);
ret=arr[input](x,y);//利用数组下标调用相应的函数
printf("ret = %d\n", ret);
}
else if (input == 0)
{
printf("退出计算器\n");
break;
}
else
{
printf("很遗憾,并没有这个选项:\n");
break;
}
} while (input);
return 0;
}
三、函数指针数组指针
函数指针数组指针:
指向一个数组.该数组的成员都是函数指针.
写法:
1.先写出函数指针
char* (*p)(const int, double, char*);
2.改成函数指针数组
char* (*pp[3])(const int, double, char*);
3.最后写成函数指针数组指针
char* (*(*ppp)[3])(const int, double, char*);
可以通过函数指针p调用函数,
也可以通过函数指针数组的元素,pp[0]调用函数.
最后,可以通过(*ppp)找到函数指针数组,(*ppp)[0]调用函数.
示例:
char* test(const int a, double b, char* c)
{
char* ret = "AAABBBCCCDDD\n";
return ret;
}
#include <stdio.h>
int main()
{
//函数指针
char* (*p)(const int, double, char*)=test;
//函数指针数组
char* (*pp[3])(const int, double, char*);
pp[0] = test;
//函数指针数组指针
char* (*(*ppp)[3])(const int, double, char*)=&pp;
printf("%s", test(0,0,0));
printf("%s", p(0, 0, 0));
printf("%s", pp[0](0, 0, 0));
printf("%s", (*ppp)[0](0, 0, 0));
return;
}
四、回调函数
回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数。回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进行响应。
大家还记得冒泡排序吗?
但是冒泡排序只能排序整形,而qsort函数,内部采用快速排序,可以排序各种类型的数据,接下来展示qsort排序部分类型的方法.
示例:qsort函数部分应用
分别将元素比较方法int_cmp和char_cmp的指针(地址) 传给 qsot函数.由qsort函数调用这些比较函数.
#include <stdlib.h>
#include <stdio.h>
int int_cmp(void* e1, void* e2)//整形元素排序方法
{
return *(int*)e1 - *(int*)e2;
}
int char_cmp(void* e1, void* e2)//字符型元素排序方法
{
return (*(char*)e1) - (*(char*)e2);
}
int main()
{
int arr1[10] = { 4,5,1,8,9,2,10,3,7,6 };
char arr2[] = "fbadegc";
int sz1 = sizeof(arr1) / sizeof(arr1[0]);
int sz2 = sizeof(arr2) / sizeof(arr2[0]);
qsort(arr1,sz1,sizeof(arr1[0]),int_cmp);
for (int i=0; i < sz1; i++)
{
printf("%d ", arr1[i]);
}
printf("\n");
qsort(arr2, sz2, sizeof(arr2[0]), char_cmp);
for (int i = 0; i < sz2; i++)
{
printf("%c ", arr2[i]);
}
printf("%s", arr2);
return 0;
}
运行结果:
1 2 3 4 5 6 7 8 9 10
a b c d e f g
qsort函数用冒泡排序的模拟实现,以及各种类型的排序.这周会加更出来的,好文不怕晚.🤣🤣🤣
最后,如果文章对大家有帮助的话,求一波三连吧!
💗💗💗886