深入解析 qsort 排序(上),它为什么是万能排序?

news2025/1/20 7:22:07

        前言:对于库函数有适当了解的朋友们,对于 qsort 函数想必是有认知的,因为他可以对任意数据类型进行排序的功能属实是有点厉害的,本次分享,笔者就给大家带来 qsort 函数的全面的解读

        本次知识的分享笔者分为上下俩卷文章进行讲解,在本篇文章中,给大家带来 qsort 函数的基础认知,以及他如何进行功能实现的背后原理讲解


目录

一.什么是 qsort 函数

二. qsort 的使用 

1.对于整形数组的排序

2.对于浮点型数组的排序 

3.对于结构体数组的排序

通过姓名首字母排序

通过年龄进行排序


一.什么是 qsort 函数

我们打开 cplusplus 网站查看详细定义

qsort - C++ Reference (cplusplus.com)

在官方的定义中是这样说的:

  • 对所指向的数组元素进行排序,每个元素长度为字节,使用函数确定顺序
  • 此函数使用的排序算法通过调用指定的函数,将指向元素的指针作为参数来比较元素对
  • 该函数不返回任何值,但通过重新排序所定义的元素来修改所指向数组的内容
  • 等效元素的顺序未定义

        翻译成我们自己的语言就是,这个函数是用来对数组元素排序的我们自己用函数来确定排序的规则 ,并且对于数组元素的类型是没有作任何要求的,这也就意味着,我们可以对任意类型的元素进行任意规则的排序

        还是看上图,这个函数有 4 个参数,并且不进行返回值,我们接下来分别解析一下具体是怎么样的一个意思:

void qsort (void* base, 
			size_t num,
			size_t size,
			int (*compar)(const void*, const void*));

 

对于这样的文本,我们还是要转化为自己好理解的方式:

  1. void* base:待排序数组的第一个元素的地址
  2. size_t num:待排序数组的元素个数
  3. size_t size:待排序数组中一个元素的大小
  4. int (*compar)(const void*, const void*):函数指针-cmp指向了一个函数,这个函数是用来比较两个元素的,e1和e2中存放的是需要比较的两个元素的地址

        另外,对于第 4 点,也就是具体比较方法的函数,它的返回值是需要注意的,只返回三类值,大于零,等于零,小于零,具体含义在后文实例中进行讲解,这里就不再多做赘述

二. qsort 的使用 

1.对于整形数组的排序

我们定义一个整型数组

	int arr1[10] = { 0,1,2,3,4,5,6,7,8,9 };

然后定义一个函数打印数组方便我们观察

void print1(int* arr)
{
	for (int i = 0; i < 10; i++)
	{
		printf("%d ", *(arr + i));
	}
	printf("\n");
}

然后我们调用 qsort 进行排序,在这里我们详细的看看四个参数:

  • 第一个参数 arr1,指向数组arr1,我们就得到了需要排序的数组的第一个元素的地址
  • 第二个参数 10,代表的是这个数组的大小是 10
  • 第三个参数 4,代表的是数组元素的字节大小,arr1 是个整形数组,整形 int 是 4 个字节,所以这里的参数是4
  • 第四个参数 comp_int,这是个函数,更是它这个函数本身的地址,我们将这里的函数地址作为参数,就可以调用我们的判断规则的函数
qsort(arr1, 10, 4, comp_int);
int comp_int(const void* a,const void* b)
{
	return *(int*)b - *(int*)a;
}

        判断函数这里更是我们需要注意的地方,我们进入到判断函数中,这个判断函数又有俩个参数,都是 void* 的类型,这样设定的目的是为了拿到一个地址,为了泛型编程的思想,我们这里只管拿到地址,不需要思考怎么处理这俩个地址,所以使用 void* 这样就能满足更多的需求,后期我们想要更改判断条件的话,也只需要更改这个函数内部的数据,不影响我们整体传参,这是一种非常高级的编程思想,我们可以进行适当的学习,同时在公司内部编程的时候,往往是很多人进行编程一个程序,这样设置更是为了方便这种团队编程的思想和功能

        进入函数体,我们首先先确定我们的具体需求,我们要的是俩个整数进行比较,但是这里我们拿到的地址是 void* 类型的,所以我们需要进行强制转化,把俩个参数转化为整形,这样才能满足我们的需求,拿到整形地址后对其解引用就可以访问里面的数据然后进行判断了,这里我们设置的是如果 b>a 就返回正值,也就是后面的数字大于前面的数字的情况下,我们才进行交换

        输出结果实例如下,我们可以看见,对于原本升序的数组,经过我们的排序后,就变成了降序

	int arr1[10] = { 0,1,2,3,4,5,6,7,8,9 };
	print1(arr1);
	qsort(arr1, 10, 4, comp_int);
	print1(arr1);

2.对于浮点型数组的排序 

         和刚才整形数组的排序类似,我们定义一个浮点型数组,然后在排序前和排序后分别进行打印一次,我们这里需要重点观察的是他的判断函数,也就是他的第四个参数

	float arr2[5] = { 1.2,3.4,5.6,7.8,9.9 };
	print2(arr2);
	qsort(arr2, 5, sizeof(float), comp_float);
	print2(arr2);

        我们观察判断函数,我们发现对于 void* 类型的转化为浮点型指针,然后进行解引用访问,我们试着运行一下看看可不可以这样写

int comp_float(const void* a, const void* b)
{
	return *(float*)b - *(float*)a;
}

        经过实验,我们发现 void* 类型强制转化为浮点型也是可以正常运行的,这里也就验证了我们刚才说过的泛型编程的思想 

 

3.对于结构体数组的排序

        刚才简单的数据类型我们都实验过了,我们好像可以发现这样设置判断函数非常的方便,参数内不管接受什么数据类型,统一写 void* 类型就可以了,那我们现在不妨试一些复杂的数据类型 —— 结构体数组

        在一切实验之前,我们先得写一个结构体类型,还有其对于的结构体数组,我们写个学生的结构体,内容由学生姓名和年龄组成

struct stu
{
	char name[20];
	int age;
};

然后对于结构体数组,我们分别如下定义,张三19岁,李四20岁,王五21岁

struct stu arr3[] ={{"zhangsan",19},{"lisi",20},{"wangswu",21}};

        做好一切准备工作后,我们进行 qsort 排序的设置,我们可以分为俩种方式排序,一种是用姓名首字母进行排序,一种是由年龄大小进行排序

通过姓名首字母排序

我们如下设置,我们来看看 qsort 的四个参数:

  • 第一个参数 arr3,指向的是结构体数组的第一个元素的地址
  • 第二个参数 3,代表的是这个结构体数组一共有 3 个元素
  • 第三个参数 sizeof(arr3[0]),代表的是结构体数组的每一个元素的大小,这一个元素包含了学生的姓名和年龄
  • 第四个参数 comp_name,代表的是排序的规则函数
	qsort(arr3, 3, sizeof(arr3[0]), comp_name);
int comp_name(const void* a, const void* b)
{
	return strcmp(((struct stu*)a)->name, ((struct stu*)b)->name);
}

        我们还是重点看一下排序规则的函数,参数方面还是和我们上面讲的一样,俩个 void* 方便我们后期修改,进入函数体,我们将俩个参数强制转化为结构体类型,然后分别指向结构体内部的姓名变量;然后我们调用了  strcmp 函数,这个函数的功能是比较俩个字符串的首元素的 ACSSII 大小,然后进行返回值,返回值和外边的比较方法的返回值的格式和大小一模一样,如下图

通过年龄进行排序

相较于上述的姓名排序,这里的年龄排序就简单的多了

	qsort(arr3, 3, sizeof(arr3[0]), comp_age);
int comp_age(const void* a, const void* b)
{
	return ((struct stu*)b)->age - ((struct stu*)a)->age;
}

        我们这里对年龄整体的处理和上述的整形数组的处理是一样的,唯一不同的就是我们拿到数据的方式,我们先将其强制转换为结构体类型,然后通过结构体类型指向的年龄进行比较


        通过了上述的实例讲述,我们明白了,对于任何的数据类型,我们在拿这个参数的时候,都是用 void* 类型进行取址,然后对于具体比较操作的时候,我们再将其进行具体的操作,强制转化为我们需要的数据类型,然后进行排序,这样就实现了其 “万能排序” 的功能

下一篇文章我们就探讨如何对 qsort 函数的模拟实现

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

[Java]JDK8新特性

一、Java版本迭代概述 1.1发布特点&#xff08;小步快跑&#xff0c;快速迭代&#xff09; 发行版本发行时间备注Java 1.01996.01.23Sun公司发布了Java的第一个开发工具包Java 5.02004.09.30①版本号从1.4直接更新至5.0&#xff1b;②平台更名为JavaSE、JavaEE、JavaMEJava 8…

一些docker笔记

一些docker笔记 docker是一个跨平台&#xff0c;可迁移的应用虚拟化,容器化服务平台Docker口号1&#xff1a;Build,Ship and Run (构建&#xff0c;发送和运行) Docker口号2: Build once,Run anywhere (构建一次&#xff0c;到处能用)docker一些概念 docker仓库 官方有dockeHu…

Linux命令200例:expr一个用于进行数值表达式求值的工具

&#x1f3c6;作者简介&#xff0c;黑夜开发者&#xff0c;CSDN领军人物&#xff0c;全栈领域优质创作者✌。CSDN专家博主&#xff0c;阿里云社区专家博主&#xff0c;2023年6月csdn上海赛道top4。 &#x1f3c6;数年电商行业从业经验&#xff0c;历任核心研发工程师&#xff0…

网站文章生成技术-网站文章生成工具免费

大家好&#xff0c;今天我想和大家分享一些关于网站文章生成的疑虑和期待。作为一个常常需要在网站上发布文章的人&#xff0c;我对这项技术的发展充满了好奇和担忧。在这篇文章中&#xff0c;我将坦率地表达我的想法&#xff0c;希望能引发一些思考。 让我谈一谈我的疑虑。网站…

Python 图形化界面基础篇:使用网格布局( Grid Layout )排列元素

Python 图形化界面基础篇&#xff1a;使用网格布局&#xff08; Grid Layout &#xff09;排列元素 引言什么是 Tkinter 的网格布局&#xff1f;步骤1&#xff1a;导入 Tkinter 模块步骤2&#xff1a;创建 Tkinter 窗口步骤3&#xff1a;创建网格步骤4&#xff1a;将元素放置在…

天线原理【1】 天线辐射的物理过程

1 前言 前面讲以振子方程入手分析电磁场问题的解的时候&#xff0c;有网友发信息说这和天线有什么关系&#xff0c;怎么从振子入手分析天线&#xff1b; 那我就开始写几次关于天线的。 有一种说法是&#xff0c;能给任何人讲懂的理论&#xff0c;才说明你真的懂了。 对天线部…

Dbeaver自动换行

Dbeaver自动换行 自己最近在使用dbeaver进行SQL语句的执行,发现,SQL语句太长不能自动换行,要拖很久,很麻烦 工具嘛,就是要顺我心意,不然用着多没意思!话不多说,上代码! 就设置下,自动换行 操作步骤 1.点击编辑 2.选择格式 3.第三步如上所示 到这一步,dbeaver的自动换行就完成,…

领域驱动设计:领域模型与代码模型的一致性

文章目录 领域对象的整理从领域模型到微服务的设计领域层的领域对象应用层的领域对象 领域对象与微服务代码对象的映射典型的领域模型非典型领域模型 DDD 强调先构建领域模型然后设计微服务&#xff0c;以保证领域模型和微服务的一体性&#xff0c;因此我们不能脱离领域模型来谈…

LeetCode(力扣)134. 加油站Python

LeetCode134. 加油站 题目链接代码 题目链接 https://leetcode.cn/problems/gas-station/description/ 代码 class Solution:def canCompleteCircuit(self, gas: List[int], cost: List[int]) -> int:cursum 0minfuel float(inf)for i in range(len(gas)):rest gas[i…

leetcode刷题_栈相关_c++版

&#xff08;1&#xff09;225用栈实现队列–简单 请你仅使用两个队列实现一个后入先出&#xff08;LIFO&#xff09;的栈&#xff0c;并支持普通栈的全部四种操作&#xff08;push、top、pop 和 empty&#xff09;。 实现 MyStack 类&#xff1a; void push(int x) 将元素 …

任务长期不释放和占用单节点持续的cpu,导致hivesever2本身内存泄漏造成

任务长期不释放和占用单节点持续的cpu&#xff0c;导致hivesever2本身内存泄漏造成 产生的原因在于&#xff1a; 查询过于复杂或者数据量过大&#xff1a;当有复杂的查询或处理大量数据的请求时&#xff0c;HiveServer2可能会出现高负载。这可能涉及大量的计算、IO操作或涉及大…

支付宝小程序排名优化,一个小白的成长手记

那是一个风和日丽的周末早上,阳光透过窗帘洒进屋内,温暖了我的双脚。这是我加入新公司的第一个周末,我坐在桌前,满怀激情地准备开发我的第一个支付宝小程序。【名即薇】 经过两天两夜的奋战,我终于完成了一个初版的支付宝小程序。是一个集美食资讯、餐厅点评、外卖订餐于一体的…

Springboot整合整合Swagger3

常用注解 Api&#xff1a;用在请求的类上&#xff0c;表示对类的说明 tags“说明该类的作用&#xff0c;可以在UI界面上看到的注解”value“该参数没什么意义&#xff0c;在UI界面上也看到&#xff0c;所以不需要配置” ApiOperation&#xff1a;用在请求的方法上&#xff0c;…

【嵌入式】化繁为简 UART、I2C、SPI整理

本文参考&#xff1a;浅谈单片机通信&#xff0c;化繁为简UART、I2C、SPI学习全家桶&#xff0c;你值得拥有&#xff01;_哔哩哔哩_bilibili 单片机的数据都是以0、1发送的&#xff0c;每一位发送多少时间取决于波特率 。 波特率是发送二进制数据位的速率&#xff0c;单位是b…

认识数据分析

文章目录 1. 认识数据分析1.1 数据自身的三大属性1.2 建数仓 数据分析的工程技术1.3 数据分析解决问题的原理1.4 数据分析的具体流程1.5 数据的中心化和智能化1.6 数据分析的四种类型和六个方向 1. 认识数据分析 1.1 数据自身的三大属性 客观&#xff1a;用数字衡量和表现一件…

vue2+three.js+blender(实现3d 模型引入并可点击效果)

2023.9.13今天我学习了如何把3d建模里面的模型引入到vue中&#xff0c;并可以实现拖动&#xff0c;点击的效果&#xff1a; 首先安装&#xff1a; npm install three 相关代码如下&#xff1a; <!--3d基础版&#xff0c;实现单个3d图形--> <template><div>&…

【服务器】ASUS ESC4000-E11 安装系统

ASUS ESC4000-E11说明书 没找到 ASUS ESC4000-E11的说明书&#xff0c;下面是ESC4000A-E11的说明书&#xff1a; https://manualzz.com/doc/65032674/asus-esc4000a-e11-servers-and-workstation-user-manual 下载地址&#xff1a; https://www.manualslib.com/manual/231379…

【PTA】PAT(甲级)2022年冬季考试自测

个人学习记录&#xff0c;代码难免不尽人意。 这次考试&#xff0c;e&#xff0c;第一题我看好多人都没理解题意做错了&#xff0c;还有第四题真的是比较意外&#xff0c;之前做的题都是给序列建树&#xff0c;但是这次让判断是否可以生成树&#xff0c;之前从来没有遇到过这类…

leecode 每日一题 2596. 检查骑士巡视方案

2596. 检查骑士巡视方案 骑士在一张 n x n 的棋盘上巡视。在 有效 的巡视方案中&#xff0c;骑士会从棋盘的 左上角 出发&#xff0c;并且访问棋盘上的每个格子 恰好一次 。 给你一个 n x n 的整数矩阵 grid &#xff0c;由范围 [0, n * n - 1] 内的不同整数组成&#xff0c;其…

pyechart练习(一):画图小练习

1、使用Map制作全球人口分布图 import math import osimport matplotlib.pyplot as plt from pyecharts.charts import Map from pyecharts import options as opts# 只有部分国家的人口数据 POPULATION [["China", 1420062022], ["India", 1368737513],…