第十二讲:指针(4)

第十二讲:指针(4)

  • 1.回调函数
    • 1.1什么是回调函数
    • 1.2深入理解并使用回调函数
      • 1.2.1简单写法
      • 1.2.2优化
  • 2.qsort函数详解
    • 2.1函数简单介绍
    • 2.3qsort函数使用举例
      • 2.3.1qsort函数排序整形数据
      • 2.3.2qsort函数排序结构数据
  • 3.qsort函数的模拟实现
    • 3.1冒泡排序
    • 3.2冒泡排序的改造

总结:该篇博客详细对以下两个内容进行讲解:
1.回调函数
2.qsort函数

1.回调函数

1.1什么是回调函数

回调函数就是一个通过函数指针调用的函数

如果我们将函数的地址作为参数传递给另一个函数,当通过这个指针被用来调用其指向的函数时,被调用的函数就是回调函数,回调函数不是由该函数的实现方法直接调用,而是在特定的事件或条件发生时由另外一方调用的,用于对该事件或条件进行响应
下面我们通过代码来理解什么是回调函数:

1.2深入理解并使用回调函数

我们写一个简易的计算器

1.2.1简单写法


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(" 0:exit \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;
}

这个方法写的十分冗余,下面我们来进行优化

1.2.2优化


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 (*pa)(int, int)) //使用函数指针变量来接受函数指针
{printf("请输入两个操作数:\n");int x, y;x = 0;y = 0;scanf("%d %d", &x, &y);int set = pa(x, y);printf("%d\n", set);
}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函数简单介绍

这是一个执行快速排序的函数,包含在<stdlib.h>库中

函数原型:


void qsort( void *base, size_t num, size_t width, int (__cdecl *compare )(const void *elem1, const void *elem2 ) );

对各个参数进行介绍如下:

1.base:指向要排序数组的首元素的地址
2.num:数组中的元素的个数
3.width:数组中每个元素的大小,单位是字节
4.int (__cdecl *compare )(const void *elem1, const void *elem2 ):
这是一个由编程者自己写的函数,将函数地址传入,函数的形式被规定,该函数定义了如何比较两个元素

对于自己定义的这个函数,通过其返回值决定所排序元素的顺序,遵循以下原则 :
在这里插入图片描述

2.3qsort函数使用举例

2.3.1qsort函数排序整形数据


//使⽤qsort函数排序整型数据int Sort1(const void* pa, const void* pb)
{return (*(int *)pa - *(int*)pb);
}int main()
{int arr[] = { 2,3,4,5,1,6,7,9,10 };int sz = sizeof(arr) / sizeof(arr[0]);qsort(arr, sz, sizeof(arr[0]), Sort1); //void qsort(void* base, //size_t num, //size_t width, //int(__cdecl * compare)(const void* elem1, const void* elem2));Print(arr, sz);return 0;
}

2.3.2qsort函数排序结构数据


//使⽤qsort函数排序结构数据//创建一个结构类型,假设为学生数据
struct Stu
{char name[20];int age;
};//按照名字排序
int SortByName(const void* pa, const void* pb)
{return (strcmp(((struct Stu*)pa)->name, ((struct Stu*)pb)->name));
}//按照年龄排序
int SortByAge(const void* pa, const void* pb)
{return (((struct Stu*)pa)->age - ((struct Stu*)pb)->age);
}int main()
{//给定学生数据struct Stu arr[] = { {"zhangsan", 20}, {"lisi", 30}, {"wangwu", 15} };int sz = sizeof(arr) / sizeof(arr[0]);//按照名字排序qsort(arr, sz, sizeof(arr[0]), SortByName); //void qsort(void* base, //size_t num, //size_t width, //int(__cdecl * compare)(const void* elem1, const void* elem2));//按照年龄排序qsort(arr, sz, sizeof(arr[0]), SortByAge); //void qsort(void* base, //size_t num, //size_t width, //int(__cdecl * compare)(const void* elem1, const void* elem2));return 0;
}

3.qsort函数的模拟实现

我们使用冒泡排序来进行qsort函数的模拟实现,首先先介绍一下什么是冒泡排序

3.1冒泡排序

目前的排序方式有很多,快速排序,希尔排序,冒泡排序,我们现在使用冒泡排序进行类比排序


void BubbleSort(int* pa, int sz)
{for (int i = 0; i < sz - 1; i++){for (int j = 0; j < sz - 1 - i; j++){if (*(pa + j) > *(pa + j + 1)){int temp = *(pa + j);*(pa + j) = *(pa + 1 + j);*(pa + 1 + j) = temp;}}}
}void Print(int* arr, int sz)
{for (int i = 0; i < sz; i++)printf("%d ", *(arr + i));
}int main()
{int arr[] = { 2,3,4,5,1,6,7,9,10 };//冒泡排序int sz = sizeof(arr) / sizeof(arr[0]);BubbleSort(arr, sz);//进行打印Print(arr, sz);return 0;
}

3.2冒泡排序的改造


//冒泡函数的改造
void Change(char* pa, char* pb, int sz)
{for(int i = 0; i<sz; i++){char temp = *pa;*pa = *pb;*pb = temp;pa++;pb++;}
}void BubbleSort(void* pa, size_t sz, size_t width, int(* compare)(const void* pa, const void* pb))
{for (int i = 0; i < sz - 1; i++)  //尽管是改造,但是这两个for循环是不用改变的{                                 //第一个for循环表示循环次数for (int j = 0; j < sz - 1 - i; j++) //第二个for循环是将两个数进行比较{//仍然是判断,如果compare函数返回值大于0,进行交换if (compare((char*)pa + j * width, (char*)pa + (j + 1) * width) > 0)Change((char*)pa + j * width, (char*)pa + (j + 1) * width, width); //进行交换,但是这个交换一次要交换数组类型个字节}}
}int main()
{//给定学生数据struct Stu arr[] = { {"zhangsan", 20}, {"lisi", 30}, {"wangwu", 15} };//冒泡排序int sz = sizeof(arr) / sizeof(arr[0]);BubbleSort(arr, sz, sizeof(arr[0]), SortByName);  //void qsort(void* base, //size_t num, //size_t width, //int(__cdecl * compare)(const void* elem1, const void* elem2));BubbleSort(arr, sz, sizeof(arr[0]), SortByAge);return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/696168.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

2024最新软件测试面试题及答案【史上最全】

以下是软件测试相关的面试题及答案&#xff0c;欢迎大家参考! 1、你的测试职业发展是什么? 测试经验越多&#xff0c;测试能力越高。所以我的职业发展是需要时间积累的&#xff0c;一步步向着高级测试工程师奔去。而且我也有初步的职业规划&#xff0c;前3年积累测试经验&…

渗透测试工具--AppInfoScanner 的安装与使用(一)

简洁 ApplicationScanner是一个快速稳定的App代码扫描工具&#xff0c;其主要功能是对ipa和apk文件进行扫描&#xff0c;以快速发现存在风险的代码。这款工具基于Python 3.7实现其主要功能&#xff0c;同时apk检测部分需要JDK 11的支持&#xff0c;因此它具备较好的跨平台特性…

Stable Diffusion WebUI 绘画

配置环境介绍​ 目前平台集成了 Stable Diffusion WebUI 的官方镜像&#xff0c;该镜像中整合如下资源&#xff1a; GpuMall智算云 | 省钱、好用、弹性。租GPU就上GpuMall,面向AI开发者的GPU云平台 Stable Diffusion WebUI版本&#xff1a;v1.7.0 Python版本&#xff1a;3.1…

娱乐营销的新玩法:Kompas.ai如何让内容更加趣味化

在数字化时代&#xff0c;内容营销已成为品牌与消费者沟通的重要桥梁。然而&#xff0c;随着信息的爆炸式增长&#xff0c;用户的注意力越来越分散&#xff0c;传统的营销方式已经难以吸引用户的兴趣。在这种背景下&#xff0c;娱乐营销应运而生&#xff0c;它通过将娱乐元素融…

忍痛分享一个标签页插件

如果你想要一个自定义收藏页面的标签页插件&#xff0c;他没有广告&#xff0c;而且支持很多插件&#xff0c;如股票、时间、倒计时、备忘录&#xff0c;那么我必须推荐你一个插件了。 itab 标签页。 标签页可以分组&#xff0c;可以自定义图标&#xff0c;很多都能自动识别 l…

VS2019正确的安装Eigen库,解决所有报错(全网最详细!!)-转

1.下载库 1.第一种方法 官方链接&#xff1a; http://eigen.tuxfamily.org/index.php?titleMain_Page https://gitlab.com/libeigen/eigen/-/releases 选择后面三个文件中的zip下载&#xff0c;解压得到 2.第二种方法 百度云链接&#xff1a;https://pan.baidu.com/s/16…

SpringBoot整合Swagger,让开发更遍历

个人简介&#xff1a;Java领域新星创作者&#xff1b;阿里云技术博主、星级博主、专家博主&#xff1b;正在Java学习的路上摸爬滚打&#xff0c;记录学习的过程~ 个人主页&#xff1a;.29.的博客 学习社区&#xff1a;进去逛一逛~ SpringBoot整合Swagger&#xff0c;让开发更遍…

鸿蒙 DevEcoStudio:用户名密码获取保存

【使用首选项实现用户名密码保存获取】 打开src/main/ets/entryability路径下的EntryAbility.ts文件 在 export default class EntryAbility extends UIAbility {onCreate(want, launchParam) {hilog.info(0x0000, testTag, %{public}s, Ability onCreate);下边添加内容&…

车牌检测识别功能实现(pyqt)

在本专题前面相关博客中已经讲述了 pyqt + yolo + lprnet 实现的车牌检测识别功能。带qt界面的。 本博文将结合前面训练好的模型来实现车牌的检测与识别。并用pyqt实现界面。最终通过检测车牌检测识别功能。 1)、通过pyqt5设计界面 ui文件如下: <?xml version="1…

分享5个免费AI写作软件

在数字化时代&#xff0c;人工智能&#xff08;AI&#xff09;正以惊人的速度渗透到我们生活的方方面面&#xff0c;而写作领域也不例外。AI写作工具的出现&#xff0c;不仅改变了传统的写作流程&#xff0c;更在创意表达、文本生成、语言校正等方面展现了其独特的优势。这些工…

软件设计中的数字:7

“ 使软件更易理解的秘密&#xff1a;米勒法则” 小游戏 学习之前先一起玩一个小游戏。 3秒钟时间&#xff0c;看看下面的图片中有多少个小块&#xff1f; 3秒到了&#xff0c;数出来了吗&#xff1f;22个。 没数出来也没关系&#xff0c;我也没数出来o(╥﹏╥)o 现在&…

【MATLAB源码-第207期】基于matlab的单相光伏并网系统仿真,并网策略采用基于扰动观测法的MPPT模型和使用电压电流双闭环SPWM控制。

操作环境&#xff1a; MATLAB 2022a 1、算法描述 本文将重点分析光伏发电最大功率点跟踪&#xff08;MPPT&#xff09;技术和逆变器的并网控制技术&#xff0c;并在Simulink环境下建立模拟系统&#xff0c;以体现这些技术的应用与效果。文章结构如下&#xff1a;首先简介光伏…