当前位置: 首页 > news >正文

【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*类型的目的是:把一大块数据切成小块交换 一个字节一个字节的交换

欢迎大家三连🎉 🎉 🎉

相关文章:

  • 智慧联络中心SaaS平台Java项目面试实战
  • Linux操作系统从入门到实战(三)Linux基础指令(上)
  • 【现代深度学习技术】循环神经网络06:循环神经网络的简洁实现
  • MySQL8的安装方法
  • 四步完成机房3D建模仿真:小白也能快速上手
  • 【Rust】Rust中的枚举与模式匹配,原理解析与应用实战
  • Pandas读取Excel文件教程:从入门到精通
  • 制作一个简单的操作系统9
  • AI 编程工具:Augment Code
  • 手机打电话时电脑坐席同时收听对方说话并插入IVR预录声音片段
  • Semantic Kernel也能充当MCP Client
  • 像拆盲盒一样读懂 XML:从新手到掌握它在大数据的用武之地
  • ​​激光雷达(LiDAR)数据​​ 的标准存储格式.las文件介绍
  • AI数字人:繁荣背后的伦理困境与法律迷局(8/10)
  • 【因果推断】(二)CV中的应用
  • 大肠杆菌诱导蛋白时OD600=0.6-0.8添加IPTG的思考-实验操作系列-009
  • Mysql日志undo redo binlog与更新一条数据的执行过程详解
  • 基于C#+Unity实现遇见李白小游戏
  • 【金仓数据库征文】金仓数据库KingbaseES:千行百业国产化征程中的璀璨之星
  • 使用yolo用python检测
  • 哈工大赵杰:人形机器人要拓展人的能力而非一味复制,未来产业要做成至少10年
  • 传染病防治法修订草案:拟加强医疗机构疾病预防控制能力建设
  • 双拥主题歌曲MV:爱我人民,爱我军
  • 《哪吒之魔童降世》电影版权方诉《仙侠神域》游戏运营方侵权案开庭
  • 马上评︱大厂取消“大小周”,会让加班隐形化吗
  • 山西一国道塌陷致2死后续:地质雷达检测出10处道路病害