qsort使用举例和qsort函数的模拟实现

qsort使用举例

qsort是C语言中的一个标准库函数,用于对数组或者其他数据结构中的元素进行排序。它的原型如下:
void qsort(void *base, size_t nmemb, size_t size, int (*compar)(const void *, const void *));

我们可以去官网搜来看一看:

那么对于其中的参数,下面也有相应的英文解释:

- base:指向要排序的数组或数据结构的第一个元素的指针。
- nmemb:要排序的元素的数量。
- size:每个元素的大小(以字节为单位)。
- compar:指向用于比较两个元素的函数的指针。

compar函数用于比较两个元素的大小关系,返回值为整数,表示两个元素的大小关系。具体返回值的含义如下:
- 若返回值小于0,则表示第一个元素小于第二个元素。
- 若返回值等于0,则表示两个元素相等。
- 若返回值大于0,则表示第一个元素大于第二个元素。

qsort函数根据compar函数的返回值对数组或数据结构中的元素进行排序,可以按照升序或降序进行排序。
那么知道以上这些,下面我们就来使用这个库函数,首先我们以排序整型数组为例:

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

假设我们要排列以上的10个元素,那么我们根据上面对于库函数qsort的认识,需要向它传入四个参数,那么第一个为- base:指向要排序的数组或数据结构的第一个元素的指针,那么就是我们的arr 了,第二个为- nmemb:要排序的元素的数量,这是我们就要计算里面有多少个元素了,我们可以这样算:

int sz = sizeof(arr) / sizeof(arr[0]);

使用第二个参数我们可以写为  sz ,第三个为- size:每个元素的大小,那么我们计算出数组中第一个元素有多少个字节就可以了 sizeof(arr[0]),最后一个为- compar:指向用于比较两个元素的函数的指针,这里我们就需要我们自己写一个函数了,告诉这个库函数qsort我们要比较什么类型的数据:

qsort(arr, sz, sizeof(arr[0]), cop_int);

那么接下来就是这个cop_int函数怎么写了,因为我们不知道会传什么样的数据,所以我们可以使用void*进行接收,返回类型为int型,因为下面是根据大于或小于或等于来排序的,而且不希望改我们所要排序的数值,所以我们这样写到:

int cop_int(const void* p1, const void* p2)

上面传来了void*类型的数据,而void*的数据不能够直接进行运算操作,所以下面我们要进行强制类型转换,作差看看两数的情况:

return *(int*)p1 - *(int*)p2;

下面就来看看我们整体的代码:

#include <stdlib.h>//头文件void Print_arr(int arr[], int sz)
{for (int i = 0; i < sz; i++){printf("%d ", arr[i]);}
}int cop_int(const void* p1, const void* p2)
{return *(int*)p1 - *(int*)p2;
}int main()
{int arr[10] = { 4,5,7,6,3,2,8,9,1,10 };int sz = sizeof(arr) / sizeof(arr[0]);qsort(arr, sz, sizeof(arr[0]), cop_int);Print_arr(arr, sz);return 0;
}

看看我们的运行效果:

 下面我们给结构体进行排序:

我们先随便写一个结构体:

struct Stu
{char name[20];int age;int score;
};//定义一个学生的名字,岁数,分数

我们只需要改我们的- compar:指向用于比较两个元素的函数的指针,我们先以学生的名字进行排序,那么结构体我们只需要转换成结构体指针就可以进行比较了,名字我们strcmp进行比较:

#include <stdlib.h>//qsort的头文件
#include <string.h>//strcmp的头文件
struct Stu
{char name[20];int age;int score;
};
int cop_Stu_by_name(const void* p1, const void* p2)
{return strcmp(((struct Stu*)p1)->name , ((struct Stu*)p2)->name);
}
int main()
{struct Stu arr[] = { {"zhangsan",18,76},{"lisi",28,65},{"wangmazi",25,79}};int sz = sizeof(arr) / sizeof(arr[0]);qsort(arr, sz, sizeof(arr[0]), cop_Stu_by_name);for (int i = 0; i < sz; i++){printf("%s %d %d\n", arr[i].name, arr[i].age,arr[i].score);}return 0;
}

看看运行的结果:

 是正确的哦!我们还可以以年龄进行比较还有分数啊,这里我就不排了,我们下面用冒泡排序来qsort函数的模拟实现!

qsort函数的模拟实现(采用冒泡排序法)

对于冒泡排序,我在前面也写到过,还不太清楚的小伙伴可以看看哦。

void BubbleSort(int arr[], int sz)
{for (int i = 0; i < sz - 1; i++)//趟数{for (int j = 0; j < sz - 1 - i; j++)//一趟冒泡排序{if (arr[j] > arr[j + 1])//需要进行改造的地方 1{int tmp = arr[j];// 需要进行改造的地方 2arr[j] = arr[j + 1];arr[j + 1] = tmp;}}}
}int main()
{int arr[] = { 5,7,9,4,3,6,8,1 };//5 7 9 4 3 6 8 1int sz = sizeof(arr) / sizeof(arr[0]);//计算有多少个元素BubbleSort(arr, sz);for (int i = 0; i < sz; i++){printf("%d ", arr[i]);}return 0;
}

首先我们先来想一想,那些地方需要我们改造,对于循环那些我们是不是就不用改了,那不会有影响。我们将会排序到不同的数据类型,所以我们对于 if (arr[j] > arr[j + 1])的比较,是不是需要我们进行改造呢,还有下面的交换变量,也是我们需要改的,既然上面的类型都被改了,下面也肯定是需要改的。

我们以结构体的年龄为例,那么函数的参数还是我们上面的那样,只是要改一下第四个参数:

bubble_sort2(arr, sz, sizeof(arr[0]), cop_Stu_by_age);

接下来就是对于bubble_sort2函数的定义了,下面传了四个参数,那我们也要用四个参数进行接收了,因为我们不知道传来的是什么类型的,所以在设置类型上,我们要考虑考虑,第一个为地址,我们用void*进行接受吧,void*都装的了,哈哈。第二个和第三个就用size_t了,最后一个像我们上面考虑的那个样:int (*cmp)(const void* p1, const void* p2)。

void bubble_sort2(void* base,size_t sz,size_t width, int (*cmp)(const void* p1, const void* p2))
{for (int i = 0; i < sz-1; i++){for (int j = 0; j < sz-1-i; j++){if (arr[j] > arr[j + 1]){int tmp = arr[j];arr[j] = arr[j + 1];arr[j + 1] = tmp;}}}
}

接着就是对于比较的改造了,我们这里构造一个新的函数进行比较,这样我们根据下面传来的类型去调用相应的,那么我们怎样找到我们的对应的元素进行比较呢?下面给给了我们首元素的地址,和每个元素的大小,我们是不是可以用起来呢,我们根据j的变化去找到元素。但是还要想到我们要用它来不同数据类型的排序,所以我们也要进行强制类型转换。我们强制转换成什么类型的数据呢?是不是char*最为合适呢,不管是什么类型,我们都可以一个一个的去进行交换:

if(cmp((char*)base + j * width, (char*)base + (j + 1) * width) > 0)

下面我们进行对于交换的改造:上面改成了那样,所以接收时也改成相应的就行了,我们以前进行交换的时候是创建了第三个变量进行交换,现在我们不知什么类型,就需要改进了。强制转换成什么类型比较合适呢?假设我们结构体的为7个字节大小,那我们是不是用char类型就可以了,不管你如何,我一个一个的进行交换就都可以实现,既然我们以这样的方式进行实现,那我们是不是应该把它每个元素的大小传进来呢:

void swap(char* a, char* b, size_t width)
{for (int i = 0; i < width; i++){char tmp = *a;*a = *b;*b = tmp;a++;b++;}
}

接下来我们就看看整体的代码:

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>struct Stu
{char name[20];int age;
};int cop_Stu_by_age(const void* p1, const void* p2)
{return ((struct Stu*)p1)->age - ((struct Stu*)p2)->age;
}int cop_int(const void* p1, const void* p2)
{return *(int*)p1 - *(int*)p2;
}void PrintArr(int arr[], int sz)
{for (int i = 0; i < sz; i++){printf("%d ", arr[i]);}
}void swap(char* a, char* b, size_t width)
{for (int i = 0; i < width; i++){char tmp = *a;*a = *b;*b = tmp;a++;b++;}
}//冒泡排序
void bubble_sort2(void* base,size_t sz,size_t width, int (*cmp)(const void* p1, const void* p2))
{for (int i = 0; i < sz-1; i++){for (int j = 0; j < sz-1-i; j++){//if (arr[j] > arr[j + 1])if(cmp((char*)base + j * width, (char*)base + (j + 1) * width) > 0){swap((char*)base + j * width, (char*)base + (j + 1) * width,width);/*int tmp = arr[j];arr[j] = arr[j + 1];arr[j + 1] = tmp;*/}}}
}int main()
{struct Stu arr[] = { {"aihua",18},{"zhanghong",21},{"kiki",15} };int sz = sizeof(arr) / sizeof(arr[0]);//年龄bubble_sort2(arr, sz, sizeof(arr[0]), cop_Stu_by_age);for (int i = 0; i < sz; i++){printf("%s %d\n", arr[i].name, arr[i].age);}return 0;
}

看看运行结果:

 那么也是可以的哟,我这里就是用其他类型进行排序了,感兴趣的话,可以去试试哦,本来还想着画图给你们梳理一下,可那个画图工具不给力😢😢😢!

愿你们冬不寒!❤❤❤

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

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

相关文章

C/C++---------------LeetCode第LCR. 024.反转链表

反转链表 题目及要求双指针 题目及要求 双指针 思路&#xff1a;遍历链表&#xff0c;并在访问各节点时修改 next 引用指向&#xff0c;首先&#xff0c;检查链表是否为空或者只有一个节点&#xff0c;如果是的话直接返回原始的头节点&#xff0c;然后使用三个指针来迭代整个…

Js中clientX/Y、offsetX/Y和screenX/Y之间区别

Js中client、offset和screen的区别 前言图文解说实例代码解说 前言 本文主要讲解JavaScript中clientX、clientY、offsetX、offsetY、screenX、screenY之间的区别。 图文解说 在上图中&#xff0c;有三个框&#xff0c;第一个为屏幕&#xff0c;第二个为浏览器大小&#xff0c…

【Git学习一】初始化仓库git init的使用和提交git add与git commit的使用

&#x1f601; 作者简介&#xff1a;一名大四的学生&#xff0c;致力学习前端开发技术 ⭐️个人主页&#xff1a;夜宵饽饽的主页 ❔ 系列专栏&#xff1a;Git等软件工具技术的使用 &#x1f450;学习格言&#xff1a;成功不是终点&#xff0c;失败也并非末日&#xff0c;最重要…

Java多线程(3)

Java多线程(3) 深入剖析Java线程的生命周期&#xff0c;探秘JVM的线程状态&#xff01; 线程的生命周期 Java 线程的生命周期主要包括五个阶段&#xff1a;新建、就绪、运行、阻塞和销毁。 **新建&#xff08;New&#xff09;&#xff1a;**线程对象通过 new 关键字创建&…

UnitTest框架

目标&#xff1a; 1.掌握UnitTest框架的基本使用方法 2.掌握断言的使用方法 3.掌握如何实现参数化 4.掌握测试报告的生成 1.定义 &#xff08;1&#xff09;框架(framework)&#xff1a;为解决一类事情的功能集合。&#xff08;需要按照框架的规定(套路) 去书写代码&…

Ubuntu16.04上安装Docker

Ubuntu16.04上安装Docker 更新 apt 包索引: sudo apt-get update安装依赖包,以便使用 HTTPS 仓库 sudo apt-get install apt-transport-https ca-certificates curl software-properties-common添加 Docker GPG 密钥 curl -fsSL https://download.docker.com/linux/ubuntu…

JavaWeb——CSS3的使用

目录 1. CSS概述 2. CSS引入方式 3. CSS颜色显示 4. CSS选择器 4.1. 元素&#xff08;标签&#xff09;选择器 4.2. id选择器 4.3. 类选择器 4.4. 三者优先级 5. 盒子模型 1. CSS概述 CSS&#xff0c;全称为“Cascading Style Sheets”&#xff0c;中文译为“层叠样式…

影刀sqlite的插入方法

影刀sqlite的插入方法 变量外面不用加‘’

【开源】基于Vue.js的开放实验室管理系统的设计和实现

项目编号&#xff1a; S 013 &#xff0c;文末获取源码。 \color{red}{项目编号&#xff1a;S013&#xff0c;文末获取源码。} 项目编号&#xff1a;S013&#xff0c;文末获取源码。 目录 一、摘要1.1 项目介绍1.2 项目录屏 二、研究内容2.1 实验室类型模块2.2 实验室模块2.3 实…

大语言模型的三阶段训练

为了训练专有领域模型&#xff0c;选择LLaMA2-7B作为基座模型&#xff0c;由于LLaMA模型中文词表有限&#xff0c;因此首先进行中文词表的扩展&#xff0c;然后进行三阶段训练&#xff08;增量预训练&#xff0c;有监督微调&#xff0c;强化学习&#xff09;。 代码将全部上传…

使用Redis实现分布式锁

Hi, I’m Shendi 使用Redis实现分布式锁 需求场景 需要使用到分布式锁的场景非常多&#xff0c;例如抢单等并发场景&#xff0c;这里举一个例子。 有一个商品&#xff0c;限量出售100个&#xff0c;一个用户下单&#xff0c;数量就减少一个&#xff0c;当剩下最后一个时&…

Verilog基础:仿真时x信号的产生和x信号对于各运算符的特性

相关阅读 Verilog基础https://blog.csdn.net/weixin_45791458/category_12263729.html?spm1001.2014.3001.5482 信号爆x也许是所有IC人的噩梦&#xff0c;满屏的红色波形常让人头疼不已&#xff0c;但x信号的产生原因却常常只有几种&#xff0c;只要遵循一定的代码规范&#…

图像分类(五) 全面解读复现ResNet

解读 Abstract—摘要 翻译 更深的神经网络往往更难以训练&#xff0c;我们在此提出一个残差学习的框架&#xff0c;以减轻网络的训练负担&#xff0c;这是个比以往的网络要深的多的网络。我们明确地将层作为输入学习残差函数&#xff0c;而不是学习未知的函数。我们提供了非…

ubuntu中用docker部署jenkins,并和码云实现自动化部署

1.部署jenkins docker network create jenkins docker run --name jenkins-docker --rm --detach \--privileged --network jenkins --network-alias docker \--env DOCKER_TLS_CERTDIR/certs \--volume jenkins-docker-certs:/certs/client \--volume jenkins-data:/var/jen…

美国服务器:全面剖析其主要优点与潜在缺点

​  服务器是网站搭建的灵魂。信息化的今天&#xff0c;我们仍需要它来为网站和应用程序提供稳定的运行环境。而美国作为全球信息技术靠前的国家之一&#xff0c;其服务器市场备受关注。那么&#xff0c;美国服务器究竟有哪些主要优点和潜在缺点呢? 优点 数据中心基础设施&a…

三十分钟学会Hive

Hive的概念与运用 Hive 是一个构建在Hadoop 之上的数据分析工具&#xff08;Hive 没有存储数据的能力&#xff0c;只有使用数据的能力&#xff09;&#xff0c;底层由 HDFS 来提供数据存储&#xff0c;可以将结构化的数据文件映射为一张数据库表&#xff0c;并且提供类似 SQL …

C语言 字符函数汇总,模拟实现各字符函数(炒鸡详细)

目录 求字符串长度 strlen 示例 模拟实现strlen 长度不受限制的字符串函数 strcpy 示例 模拟实现strcpy strcat 模拟实现strcat strcmp 示例 模拟实现strcmp 长度受限制的字符串函数介绍 strncpy 示例 模拟实现strncpy strncat 示例 模拟实现strncat s…

OpenCV C++ 图像处理实战 ——《OCR字符识别》

OpenCV C++ 图像处理实战 ——《OCR字符识别》 一、结果演示二、tesseract库配置2.1下载编译三、OCR字符识别3.1 文本检测方式3.1.1 RIL_BLOCK3.1.2 RIL_PARA3.1.3 RIL_TEXTLINE3.1.4 RIL_WORD3.1.5 RIL_SYMBOL3.2 英文文本检测3.3 中英文本检测四、源码测试图像下载总结一、结…

随机过程-张灏

文章目录 导论随机过程相关 学习视频来源&#xff1a;https://www.bilibili.com/video/BV18p4y1u7NP?p1&vd_source5e8340c2f2373cdec3870278aedd8ca4 将持续更新—— 第一次更新&#xff1a;2023-11-19 导论 教材&#xff1a;《随机过程及其应用》陆大絟.张颢 参考&…