C语言之深入指针及qsort函数(五)(详解介绍)

C语言之深入指针

在这篇博客看不懂的可以看看这篇C语言之深入指针(四)在上篇博客中介绍了:

  1. 函数指针变量
  2. 函数指针数组
  3. 简易计算器的实现\

文章目录

  • C语言之深入指针
    • 1 回调函数
    • 2 qsort函数的使用
      • 2.1 使用冒泡排序排序整型数组
      • 2.2 使用qsort函数排序整型数组
      • 2.2 使用qsort函数排序结构体数组
        • 2.2.1 按照年龄来排序
        • 2.2.2 按照名字来排序
      • 3 qsort函数总结

1 回调函数

使用回调函数修改前的代码:

#include <stdio.h>void menu()
{printf("********************************\n");printf("*******   1. Add   2. Sub  *****\n");printf("*******   3. Mul   4. Div  *****\n");printf("*******   0. exit         ******\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 Div(int x, int y)
{return x / y;
}
int main(){int input = 0;int x = 0;int y = 0;int ret = 0;do{menu();printf("请选择:>>");scanf("%d", &input);switch (input){case 1:printf("请输入操作数:>>");scanf("%d %d", &x, &y);ret = Add(x, y);printf("%d\n", ret);break;case 2:printf("请输入操作数:>>");scanf("%d %d", &x, &y);ret = Sub(x, y);printf("%d\n", ret);break;case 3:printf("请输入操作数:>>");scanf("%d %d", &x, &y);ret = Mul(x, y);printf("%d\n", ret);break;case 4:printf("请输入操作数:>>");scanf("%d %d", &x, &y);ret = Div(x, y);printf("%d\n", ret);break;case 0:printf("退出计算器\n");break;default:printf("请重新输入\n");}} while (input);return 0;
}
case 1:printf("请输入操作数:>>");scanf("%d %d", &x, &y);ret = Add(x, y);printf("%d\n", ret);break;

在上述代码中,可以看到在每个case语句中,代码基本相似,基本逻辑是一致的,但是输入和输出部分完全一致这么写显得代码很冗余,那么我们可以使用回调函数来减少代码的重复

回调函数:回调函数就是⼀个通过函数指针调⽤的函数
如果你把函数的指针(地址)作为参数传递给另⼀个函数,当这个指针被⽤来调⽤其所指向的函数时,被调⽤的函数就是回调函数。回调函数不是由该函数的实现⽅直接调⽤,⽽是在特定的事件或条件发⽣时由另外的⼀⽅调⽤的,⽤于对该事件或条件进⾏响应

使用回调函数修改后的代码:

#include <stdio.h>void menu()
{printf("********************************\n");printf("*******   1. Add   2. Sub  *****\n");printf("*******   3. Mul   4. Div  *****\n");printf("*******   0. exit         ******\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 Div(int x, int y)
{return x / y;
}void cacl(int (*p) (int, int) )
{int x = 0;int y = 0;int ret = 0;printf("请输入操作数:>>");scanf("%d %d", &x, &y);ret = p(x, y);printf("%d\n", ret);
}
int main(){int input = 0;do{menu();printf("请选择:>>");scanf("%d", &input);switch (input){case 1:cacl(Add);break;case 2:cacl(Sub);break;case 3:cacl(Mul);break;case 4:cacl(Div);break;case 0:printf("退出计算器\n");break;default:printf("请重新输入\n");break;}} while (input);return 0;
}

在上述代码中,使用了回调函数 cacl ,将函数的地址传给了回调函数,在回调函数中使用了加法等函数,减少了代码的重复
cacl函数的形参部分为函数指针类型用来接收函数的地址

2 qsort函数的使用

2.1 使用冒泡排序排序整型数组

给定一个整型数组,要求将其排序
最简单的方法就是使用冒泡排序
代码如下:

#include <stdio.h>
void bobble_sort(int arr[], int sz)
{int i = 0;int tmp = 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]){tmp = arr[j];arr[j] = arr[j + 1];arr[j + 1] = tmp;}}}
}Print(int arr[], int sz)
{int i = 0;for (i = 0; i < sz; i++){printf("%d ", arr[i]);}printf("\n");
}
int main(){int arr[] = { 1,4,7,2,5,8,10,3,6,9 };int sz = sizeof(arr) / sizeof(arr[0]);Print(arr, sz);bobble_sort(arr, sz);Print(arr, sz);return 0;
}

2.2 使用qsort函数排序整型数组

在2.1中使用冒泡排序排序了整型数组,但是bobble_sort这个函数只能排序整型数组,在C语言中有这么一个函数——qsort可以排序任意类型的数组

void qsort (void* base,size_t num, size_t size,            int (*compar)(const void*,const void*));//qsort函数的声明

在这里插入图片描述
(图片转载至https://cplusplus.com/)
在cplusplus中介绍到,qsort函数有四个形参

  1. void* base   //base指向的数组中第一个的元素  void* 为泛型指针可以接收任意类型的指针
  2. size_t num  //base指向的数组中元素的个数(待排序数组中的元素个数)
  3. size_t size  //base指向的数组中元素的大小(单位是字节)
  4. int (compar)(const void,const void*)) //函数指针 - 指针指向的函数是用来比较数组中的2个元素的
  5. 4中函数指针的例子:int compar (const void* p1, const void* p2);

我将在代码中详细介绍:

#include <stdio.h>
#include <stdlib.h>    //使用qsort需要包含的头文件//int compar (const void* p1, const void* p2);
int sort_intarr(const void* p1, const void* p2)  //比较数组中两个元素 形参类型是固定的
{return (*(int*)p1 - *(int*)p2);
}Print(int arr[], int sz)  //打印输出
{int i = 0;for (i = 0; i < sz; i++){printf("%d ", arr[i]);}printf("\n");
}
int main()
{int arr[] = { 1,4,7,2,5,8,10,3,6,9 };int sz = sizeof(arr) / sizeof(arr[0]);   //计算数组元素个数Print(arr, sz);  //排序前打印一次qsort(arr,sz,sizeof(arr[0]), sort_intarr);Print(arr, sz); //排序前打印一次后return 0;
}
qsort(arr,sz,sizeof(arr[0]), sort_intarr);

一、
在上述代码中传给了qsort函数4个参数
1 arr为数组名,即传给了qsort第一个元素
2 sz为计算出的元素个数
3 sizeof(arr[0])为计算出一个元素的大小
4 sort_intarr为自己写的一个函数,用于比较两个元素

int sort_intarr(const void* p1, const void* p2)
{return (*(int*)p1 - *(int*)p2);
}

二、
在上述代码中定义了一个sort_intarr函数用来比较两个元素的大小
return ( * (int * )p1 - (int * )p2);
1 由于形参为void*类型的数据,需要转换成int类型的数据进行比较

TIPS:如果想降序排列可以将return ( * (int * )p1 - * (int * )p2);改为return ( * (int * )p2 - * (int * )p1);

2.2 使用qsort函数排序结构体数组

由于结构体中的数据类型很多,所以我们得按需求来实现函数

2.2.1 按照年龄来排序

代码如下:

#include <stdio.h>
#include <stdlib.h>struct Stu
{char name[20];int age;
};int cmp_struct_by_age(const void* p1, const void* p2)
{return ((struct Stu*)p1)->age - ((struct Stu*)p1)->age;//return (*(struct Stu*)p1).age - (*(struct Stu*)p1).age;//两段代码等价
}int main()
{struct Stu arr[] = { {"zhangsan",25},{"lisi",18} ,{"wangwu",30} };int sz = sizeof(arr) / sizeof(arr[0]);qsort(arr, sz, sizeof(arr[0]), cmp_struct_by_age);return 0;
}

一、
与排序整型数组一样,传入的参数都是那4种

1 arr为数组名,即传给了qsort第一个元素
2 sz为计算出的元素个数
3 sizeof(arr[0])为计算出一个元素的大小
4 cmp_struct_by_age为自己写的一个函数,用于比较两个元素

//两段代码等价
return ((struct Stu*)p1)->age - ((struct Stu*)p1)->age;
return (*(struct Stu*)p1).age - (*(struct Stu*)p1).age;

二、
1 由于是结构体数组,所以要将void*类型的数据强制转换成结构体类型
2 如果使用 ( -> ) 操作符结构体数组不使用解引用操作符来使用
3 要想使用 ( . ) 操作符的话,需要在前面加上解引用操作符

2.2.2 按照名字来排序

代码如下:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>   //使用strcmp函数需要包含的头文件struct Stu
{char name[20];int age;
};int cmp_struct_by_name(const void* p1, const void* p2)
{return strcmp(((struct Stu*)p1)->name, ((struct Stu*)p2)->name);//return strcmp((*(struct Stu*)p1).name, (*(struct Stu*)p2).name);//两段代码等价
}int main()
{struct Stu arr[] = { {"zhangsan",25},{"lisi",18} ,{"wangwu",30} };int sz = sizeof(arr) / sizeof(arr[0]);qsort(arr, sz, sizeof(arr[0]), cmp_struct_by_name);return 0;
}

一、
与排序整型数组一样,传入的参数都是那4种

1 arr为数组名,即传给了qsort第一个元素
2 sz为计算出的元素个数
3 sizeof(arr[0])为计算出一个元素的大小
4 cmp_struct_by_name为自己写的一个函数,用于比较两个元素

//两段代码等价
return strcmp(((struct Stu*)p1)->name, ((struct Stu*)p2)->name);
return strcmp((*(struct Stu*)p1).name, (*(struct Stu*)p2).name);

二、
两个字符串之间的比较不是比较字符串长度,而是比较字符的ASCII
例如:

char str1 = "abcdef";
char str2 = "abz";

第一个字母都是a相等的话会比较下一对字符,c的ASCII值小于g的ASCII,所以str2的长度大于str1,我们可以使用strcmp函数来比较字符串的大小

3 qsort函数总结

1 qsort可以排序任意类型的数据
2 使用qsort需要包含头文件 <stdlib.h>
3 使用qsort需要传4个参数
4 根据需要排序的数据传递不同的函数 按照参数和返回值实现
5 升序使用第一个形参 - 第二个形参 降序则反之

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

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

相关文章

使用Sqoop命令从Oracle同步数据到Hive,修复数据乱码 %0A的问题

一、创建一张Hive测试表 create table test_oracle_hive(id_code string,phone_code string,status string,create_time string ) partitioned by(partition_date string) ROW FORMAT DELIMITED FIELDS TERMINATED BY ,; 创建分区字段partition_date&#xff0c…

mysqlbinlog使用记录

首先要确认mysql启用了binlog功能。一般默认启用。 mysql> select log_bin; ----------- | log_bin | ----------- | 1 | ----------- 然后确认binlog目录 mysql> select log_bin_basename; ---------------------------- | log_bin_basename | -----…

xlua源码分析(三)C#访问lua的映射

xlua源码分析&#xff08;三&#xff09;C#访问lua的映射 上一节我们主要分析了lua call C#的无wrap实现。同时我们在第一节里提到过&#xff0c;C#使用LuaTable类持有lua层的table&#xff0c;以及使用Action委托持有lua层的function。而在xlua的官方文档中&#xff0c;推荐使…

<b><strong>,<i><em>标签的区别

1. b标签和strong标签 b标签&#xff1a;仅仅是UI层面的加粗样式&#xff0c;并不具备HTML语义 strong标签&#xff1a;不仅是在UI层面的加粗样式&#xff0c;具备HTML语义&#xff0c;表示强调 2. i标签和em标签 i 标签&#xff1a;仅仅是UI层面的斜体样式&#xff0c;并不…

Django学习日志08

如何开启事务 事务的目的&#xff1a;为了保证多个SQL语句执行成功&#xff0c;执行失败&#xff0c;前后保持一致&#xff0c;保证数据安全 ACID属性&#xff1a; A&#xff1a;原子性&#xff08;Atomicity&#xff09;&#xff1a;指事务是原子的&#xff0c;对事务中的操…

系统设计之通讯协议

一、通讯协议 架构风格定义了应用程序编程接口 (API) 的不同组件如何相互交互。因此&#xff0c;它们通过提供设计和构建 API 的标准方法来确保效率、可靠性以及与其他系统集成的便捷性。以下是最常用的样式&#xff1a; 1. SOAP 成熟、全面、基于XML 最适合于企业应用 可扩展…

STM32获取最大堆栈空间

参考 stackflow相关讨论 原理 通过参考链接&#xff0c;可知探测Stack的最大深度是先在stack中填充不常用的特定值&#xff0c;然后实时检测这些值哪些发生了变化&#xff0c;变化的表示使用到了这个空间&#xff0c;如果程序完全遍历后&#xff0c;有些值还是没变&#xff…

网络运维与网络安全 学习笔记2023.11.19

网络运维与网络安全 学习笔记 第二十天 今日目标 STP工作原理、STP高级配置、MSTP工作原理 MSTP配置案例、MSTP负载均衡 STP工作原理 单点故障 PC之间的互通链路仅仅存在1个 任何一个设备或链路出现问题&#xff0c;PC之间都会无法通信 解决方案 增加冗余/备份设备 增加冗…

9 HDFS架构剖析

问题 100台服务器&#xff0c;存储空间单个200GB 20T 5T文件如何存储&#xff1f; 128MB一块 128MB81GB 1288*10241TB 5T数据分成的128MB的块数 8192 * 5 客户端(client)代表用户通过与namenode和datanode交互来访问整个文件系统。 HDFS集群有两类节点&#xff1a; 一个na…

成为电车销量的“中坚力量”,微小型车不能只有“低价”?

日常交通中&#xff0c;越来越多的汽车开始“绿牌出行”&#xff0c;市场的最新销量也不断验证着新能源车抢占更多市场的事实。 11月初&#xff0c;国内多家车企公布10月销量数据&#xff0c;其中新能源汽车销量增长仍然亮眼。根据中国工业和信息化部数据&#xff0c;我国汽车…

Nessus扫描结果出现在TE.IO或者ES容器结果查看问题解决方案

Nessus扫描结果出现在TE.IO或者ES容器结果查看问题解决方案 也是昨天晚上折腾了一个晚上到凌晨四点多,实在没有头绪,在论坛,贴吧,各种求助查贴,没有什么人解决.后面请教了一个安全圈的大佬朋友给解决了. 我的问题是在kali上的,所以只写了kali 的解决方案: 修改插件: vim /opt/…

Hadoop-- hdfs

1、HDFS中的三个进程&#xff1a;NameNode&#xff08;NN&#xff09;、DataNode(DN)、SecondNameNode(SNN) 2、NameNode&#xff08;NN&#xff09; 1、作用&#xff1a; 1、接收客户端的一个读、写的服务&#xff0c;在namenode上存储了数据文件和datanode的映射的关系。 …

【Q1—45min】

1.epoll除了边沿触发还有什么&#xff1f;与select区别. epoll 是Linux平台下的一种特有的多路复用IO实现方式&#xff0c;与传统的 select 相比&#xff0c;epoll 在性能上有很大的提升。 epoll是一种当文件描述符的内核缓冲区非空的时候&#xff0c;发出可读信号进行通知&…

划片机新手教程:从准备工作到注意事项全解析!

随着科技的飞速发展&#xff0c;划片机已成为半导体行业不可或缺的一部分。对于新手来说&#xff0c;如何正确操作划片机显得尤为重要。以下是新手操作划片机的步骤和建议。 一、准备工作 在开始操作划片机之前&#xff0c;首先需要准备好以下工具和材料&#xff1a; 1. 划片机…

Java(三)(static,代码块,单例设计模式,继承)

目录 static 有无static修饰的成员变量 有无static修饰的成员方法 static的注意事项 代码块 静态代码块 实例代码块 单例设计模式 饿汉式单例写法 懒汉式单例写法 继承 基本概念 注意事项 权限修饰符 单继承 object 方法重写 子类方法中访问其他成员(成员变量…

Flume学习笔记(3)—— Flume 自定义组件

前置知识&#xff1a; Flume学习笔记&#xff08;1&#xff09;—— Flume入门-CSDN博客 Flume学习笔记&#xff08;2&#xff09;—— Flume进阶-CSDN博客 Flume 自定义组件 自定义 Interceptor 需求分析&#xff1a;使用 Flume 采集服务器本地日志&#xff0c;需要按照日志…

交易机器人-微信群通知

微信公众号:大数据高性能计算 1 背景 背景是基于人工去做交易本身无法做到24小时无时无刻的交易,主要是虚拟币本身它是24小时交易,人无法做到24小时盯盘,其次就是如果你希望通过配置更加复杂的规则甚至需要爬取最新的信息走模型进行量化交易的时候,就需要自己去做一些量化…

Linux 环境搭建

✨个人主页&#xff1a; Anmia.&#x1f389;所属专栏&#xff1a; C Language &#x1f383;操作环境&#xff1a; Visual Studio 2019 版本 本章概要 1. 认识 Linux, 了解 Linux 的相关背景 2. 学会如何使用云服务器 3. 掌握使用远程终端工具 xshell 登陆 Linux 服务器 1. Li…

语义检索系统【全】:基于milvus语义检索系统指令全流程-快速部署版

搜索推荐系统专栏简介:搜索推荐全流程讲解(召回粗排精排重排混排)、系统架构、常见问题、算法项目实战总结、技术细节以及项目实战(含码源) 专栏详细介绍:搜索推荐系统专栏简介:搜索推荐全流程讲解(召回粗排精排重排混排)、系统架构、常见问题、算法项目实战总结、技术…

学生邮箱白嫖/免费安装JetBrains全家桶(IDEA/pycharm等) —— 保姆级教程

&#x1f9f8;欢迎来到dream_ready的博客&#xff0c;&#x1f4dc;相信您对博主首页也很感兴趣o (ˉ▽ˉ&#xff1b;) 博主首页&#xff0c;更多redis、java等优质好文以及各种保姆级教程等您挖掘&#xff01; 目录 前言 JetBrains全家桶介绍 申请过程&#xff1a; 获取学…