【C/C++】深入理解指针(五)
文章目录
- 深入理解指针(五)
- 1.回调函数是什么?
- 2.qsort使用举例
- 2.1 使用qsort函数排序整型数据
- 强调
- 2.2 使用qsort排序结构数据
- 3.qsort函数的模拟实现
深入理解指针(五)
1.回调函数是什么?
回调函数就是⼀个通过函数指针调⽤的函数。
如果你把函数的指针(地址)作为参数传递给另⼀个函数,当这个指针被⽤来调⽤其所指向的函数时,被调用的函数就是回调函数。回调函数不是由该函数的实现⽅直接调⽤,⽽是在特定的事件或条 件发⽣时由另外的⼀⽅调⽤的,⽤于对该事件或条件进⾏响应。
在深入理解指针(四)**(可以移步主页)**中我们写的计算机的实现的代码中,红⾊框中的代码是重复出现的,其中虽然执⾏计算的逻辑 是区别的,但是输⼊输出操作是冗余的,有没有办法,简化⼀些呢?
可以将相似代码抽象成函数
因为红⾊框中的代码,只有调⽤函数的逻辑是有差异的,我们可以把调⽤的函数的地址以参数的形式 传递过去,使⽤函数指针接收,函数指针指向什么函数就调⽤什么函数,这⾥其实使⽤的就是回调函 数的功能。
有了函数指针后,函数的调用可以使用函数名调用,也可以使用函数指针来调用
//使⽤回调函数改造前
#include <stdio.h>
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, y;int input = 1;int ret = 0;do{printf("*************************\n");printf(" 1:add 2:sub \n");printf(" 3:mul 4:div \n");printf("*************************\n");printf("请选择:");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>
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;
}
void calc(int(*pf)(int, int))
{int ret = 0;int x, y;printf("输⼊操作数:");scanf("%d %d", &x, &y);ret = pf(x, y);printf("ret = %d\n", ret);
}
int main()
{int input = 1;do{printf("*************************\n");printf(" 1:add 2:sub \n");printf(" 3:mul 4:div \n");printf("*************************\n");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");break;}} while (input);return 0;
}
2.qsort使用举例
2.1 使用qsort函数排序整型数据
qsort是库函数,可以排序任意类型的数据,底层使用的是快速排序的方式(头文件为(<stdlib.h>)
下边对qsort个参数进行解释
void qsort(void* base,//指针,指向的是待排序的数组的第一个元素size_t num,//base指向的待排序数组的元素的个数size_t size,//base指向待排序数组元素的大小int(*compar)(const void*, const void*)//函数指针
);
qosrt 函数有实现者
qosrt 函数的使用者 —-明确的知道要排序的是什么数据,这些数据应该如何比较,所以提供两个元素的比较函数
在深入理解指针(三)**(可以移步主页查看详情)**中,我们提到了冒泡排序,代码如下
#include<stdio.h>
void bubble_sort(int arr[], int sz)
{int i = 0;for (i = 0; i < sz - 1; i++){int j = 0;for (j = 0; j < sz - 1 - i; j++){if (arr[j] > arr[j + 1]){int tmp = arr[j + 1];arr[j + 1] = arr[j];arr[j] = tmp;}}}
}
void print_arr(int arr[], int sz)//参数写成数组的形式,本质上还是指针
//形参的数组是不会再单独创建数组空间的,所以形参的数组是可以省略掉数组大小不写的
{int i = 0;for (i = 0; i < sz; i++){printf("%d ", arr[i]);}
}
int main()
{int arr[] = { 9,8,7,6,5,4,3,2,1 };//排为升序int sz = sizeof(arr) / sizeof(arr[0]);bubble_sort(arr, sz);print_arr(arr, sz);
}
很容易发现,该代码只适用于整数
那么在使用冒泡排序的情况下,要改造函数让它能够排序任意类型的数据,该如何做到
问题的主要原因是
两个整型元素可以直接用>比较,但是两个字符串、两个结构体是不能用>比较的
参数以及数据的交换也需要更改
我们可以把两个元素的比较封装成函数,然后把函数地址传给排序函数
#include <stdio.h>
//qosrt函数的使⽤者得实现⼀个⽐较函数
int int_cmp(const void* p1, const void* p2)
{//强制类型转换if(*(int *)p1>*(int *)p2)//此处不能写成if(*p1>*p2) 因为void*类型的指针是无具体类型的指针,这种类型的指针不能直接解引用,也不能+-整数运算return 1;else if(*(int *)p1==*(int *)p2)return 0;elsereturn -1;
}
int main()
{int arr[] = { 1, 3, 5, 7, 9, 2, 4, 6, 8, 0 };int i = 0;qsort(arr, sizeof(arr) / sizeof(arr[0]), sizeof(int), int_cmp);for (i = 0; i < sizeof(arr) / sizeof(arr[0]); i++){printf("%d ", arr[i]);}printf("\n");return 0;
}
强调
void*类型的指针是无具体类型的指针,这种类型的指针不能直接解引用,也不能±整数运算
最后我们对代码做一下简化
int int_cmp(const void* p1, const void* p2)
{return (*(int*)p1 - *(int*)p2);
}
int main()
{int arr[] = { 1, 3, 5, 7, 9, 2, 4, 6, 8, 0 };int i = 0;qsort(arr, sizeof(arr) / sizeof(arr[0]), sizeof(int), int_cmp);for (i = 0; i < sizeof(arr) / sizeof(arr[0]); i++){printf("%d ", arr[i]);}printf("\n");return 0;
}
2.2 使用qsort排序结构数据
这⾥需要补充介绍结构指针和结构体成员访问操作符 ->;
结构体变量.成员名
结构体指针.成员名
结构体补充:
struct Stu//学生
{char mane[20];//名字int age;//年龄
};
int main()
{struct Stu s={"zhangsan",18};printf("%s %d",s.name,s.age);return 0;
}
void print(struct Stu* ps)
{//printf("%s %d",(*ps).name,(*ps).age); //可以,但是啰嗦printf("%s %d",ps->name,ps->age);//这两者等价
}
struct Stu//学生
{char mane[20];//名字int age;//年龄
};
int main()
{struct Stu s={"zhangsan",18};print(&s);return 0;
}
这两个结构体怎么比
1.按照名字比较————-字符串比较—–strcmp(头文件为<string.h>)
2.按照年龄比较————-整型比较
那strcmp函数是怎么比较字符串大小的呢
按照对应字符串中字符的ASII码值比较
上图中,两个字符串前两个相同,然后比较第三个字符的ASII码值
所以上边的字符串大
struct Stu //学⽣
{char name[20];//名字 int age;//年龄
};
//假设按照年龄来⽐较
int cmp_stu_by_age(const void* e1, const void* e2)
{return ((struct Stu*)e1)->age - ((struct Stu*)e2)->age;//跟之前一样,这里同样需要强制类型转换
}
//strcmp - 是库函数,是专⻔⽤来⽐较两个字符串的⼤⼩的
//假设按照名字来⽐较
int cmp_stu_by_name(const void* e1, const void* e2)
{return strcmp(((struct Stu*)e1)->name, ((struct Stu*)e2)->name);
}
//按照年龄来排序
void test2()
{struct Stu s[] = { {"zhangsan", 20}, {"lisi", 30}, {"wangwu", 15} };int sz = sizeof(s) / sizeof(s[0]);qsort(s, sz, sizeof(s[0]), cmp_stu_by_age);
}
//按照名字来排序
void test3()
{struct Stu s[] = { {"zhangsan", 20}, {"lisi", 30}, {"wangwu", 15} };int sz = sizeof(s) / sizeof(s[0]);qsort(s, sz, sizeof(s[0]), cmp_stu_by_name);
}
int main()
{test2();test3();return 0;
}
3.qsort函数的模拟实现
使⽤回调函数,模拟实现qsort(采⽤冒泡的⽅式)。
注意:这⾥第⼀次使⽤ void* 的指针,讲解 void* 的作⽤。
#include <stdio.h>
int int_cmp(const void* p1, const void* p2)//void接收任意类型指针
{return (*(int*)p1 - *(int*)p2);
}
void _swap(void* p1, void* p2, int size)
{int i = 0;for (i = 0; i < size; i++){//char tmp = *((char*)p1 + i);//*((char*)p1 + i) = *((char*)p2 + i);//*((char*)p2 + i) = tmp;char tmp=p1;//两种写法等价*p1=*p2;*p2=tmp;p1++;p2++; }
}
void bubble(void* base, int count, int size, int(*cmp)(void*, void*))
{int i = 0;int j = 0;for (i = 0; i < count - 1; i++){for (j = 0; j < count - i - 1; j++){if (cmp((char*)base + j * size, (char*)base + (j + 1) * size) > 0)//size是宽度 *size可以跳过1个数据{_swap((char*)base + j * size, (char*)base + (j + 1) * size, size);}}}
}
int main()
{int arr[] = { 1, 3, 5, 7, 9, 2, 4, 6, 8, 0 };int i = 0;bubble(arr, sizeof(arr) / sizeof(arr[0]), sizeof(int), int_cmp);for (i = 0; i < sizeof(arr) / sizeof(arr[0]); i++){printf("%d ", arr[i]);}printf("\n");return 0;
}
上述代码中
base相当于arr[0]
在swap函数中
强制转换成char*类型的目的是:把一大块数据切成小块交换 一个字节一个字节的交换
欢迎大家三连🎉 🎉 🎉