C语言标准库函数qsort( )——数据排序

简介: 大家好!我是保护小周ღ,本期为大家带来的是深度解剖C语言标准库函数 qsort(),qsort()函数他可以对任意类型的数据排序,博主会详细解释函数使用方法,以及使用快速排序的左右指针法模拟实现函数功能,这样的排序确定不来学习一下吗???

 image.gif编辑

大家好!我是保护小周ღ,本期为大家带来的是深度解剖C语言标准库函数 qsort(),qsort()函数他可以对任意类型的数据排序,博主会详细解释函数使用方法,以及使用快速排序的左右指针法模拟实现函数功能这样的排序确定不来学习一下吗???

image.gif编辑

image.gif编辑

目录

一、qsort()函数简介

二、qsort() 函数的参数

三、qsort() 函数的使用

3.1 对整型数据排序

3.2 对结构体类型数据排序

四、快速排模拟实现qsort()函数


一、qsort()函数简介

qsort() 函数是C语言标准库提供的排序函数,q==Quick,函数内部是以快速排序的思想实现的,那qsort() 函数的意义是什么呢?内部居然还使用别的排序的思想。因为常规排序是写死的,假如原先是对整型数据的排序,现在要对结构体类型的数据排序,则需要修改函数参数,函数内部数据也要相应的修改。而qsort()函数他可以对任意类型的数据排序,比如说,可以直接排整型数据,也可以排结构体类型的数据,甚至是字符串数据,通用性极强。这样的排序确定不来学习一下吗???


二、qsort() 函数的参数

image.gif编辑

很明显 qsort()函数具有4个参数,接下来博主来解剖一下这些参数代表着什么意思。

1.   void * base : 首先来了解一下什么是 void* ,这个是无具体类型的指针,void * 的指针是非常宽容的,可以接收任意类型的数据。常常用来临时存放数据,等到需要使用数据时,我们必须要强制类型转换成某一具体类型的数据,才能对数据进行操作。

image.gif编辑

对void *pa,接收了一个整型 a 的地址,我们对指针pa 进行强制类型转换(int*),再解引用 pa即可对变量a 进行操作。

void *base 存的就是待排序数据的起始地址(不能直接访问)。

这个参数是 qsort() 函数能够对任意数据排序的基础。

2. size_t  num :记录待排序数据元素个数。

3. size_t  size  : 记录待排序数据任意一个元素的所占的字节数(元素的大小)。

4. int  (*compar) (const  void*  , const  void* ) :

这其实是一个函数类型的指针,可以用来存储函数的地址,然后也提前声明了函数的参数,返回值

返回值是 int 类型,参数部分是两个 void * 类型的接收。这个函数的作用是来比较两个参数的大小,然后返回比较果结,怎么比呢? 如果是整型数据使两个参数相减,返回结果。如果是字符串,我们可以使用   strcmp(“字符串”,“字符串”);strcmp 函数的返回值也是整型数据(这个是根据对应的场景,选择比较方式),即可得到相应的结果。

这第四个参数需要我们自己设计实现,函数的作用就是比较任意两个参数,返回一个整型数据,就可以利用这个数据来判断两个元素大小,所以这是个比较排序。

image.gif编辑


三、qsort() 函数的使用

qsort()函数包含在 stdlib.h库里,所以我在使用前需要申明一下。

3.1 对整型数据排序

#include<stdio.h>
#include<stdlib.h>//头文件声明
//判断两个元素的大小
int Compar(void const *p1,void const *p2)//无具体类型的指针接收数据,使用时强制类型转换。
{
    //两个整型数据相减,即可即可得到三种信息>,<,=
    return (*(int*)p1 - *(int*)p2) *(-1); //利用乘-1改变符号控制升序还是降序
}
//打印
void Print(int *arr,int n)
{
    for (int i=0;i<n;++i)
    {
        printf("%d ",arr[i]);
    }
}
int main()
{
    int arr[] = { 8,6,4,2,3,7,1,2,3,10,9 };
    int len = sizeof(arr) / sizeof(arr[0]);//记录数组元素个数
    int size = sizeof(arr[0]);//记录某一个元素的大小所占的字节数
    //库函数
    qsort(arr, len, size, Compar);
    //第四个参数,直接传函数的地址,函数名代表函数的地址,由函数指针接收。
    //打印
    Print(arr, len);
    return 0;
}

image.gif

image.gif编辑


3.2 对结构体类型数据排序

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
typedef struct Student//定义一个Student类型的结构体
{
  char name[12]; //姓名
  int age;       //年龄
  char StuID[8]; //学号
}Student;//typedef,重命名一下,简化。
//比较任意两个元素的大小。
int CmpSort(const void* p1, const void * p2)
{
  //return ( ((Student*)p1)->age - ((Student*)p2)->age );//根据年龄来比
  return (strcmp(((Student*)p1)->name,((Student*)p2)->name));//按照姓名的首字母来比较
}
//打印
void Print(Student* ps,int n)
{
  for (int i=0;i<n;++i)
  {
    printf("%s %d %d\n",ps[i].name,ps[i].age,ps[i].StuID);
  }
}
int main()
{
  Student student[3] = { {"张三",18,"21933321"},{"李四",20,"21933323"},{"王老五",19,"21933322"}};
  int len = sizeof(student) / sizeof(student[0]);//统计Student 类型,student数组的元素的个数
  int size = sizeof(student[0]);//统计某一个Student 元素所占的字节数。
  //库函数
  qsort(student,len,size,CmpSort);
  Print(student,len);//打印
  return 0;
}

image.gif

image.gif编辑


四、快速排模拟实现qsort()函数

好了经过以上三节内容的铺垫,相信大家应该对 qsort() 函数的用法,功能明白了,接下来我们就要来模拟实现函数内部。上文说到排序思想是用快速排序的思想。那博主今天就用快速排序——左右指针法来模拟实现挖坑法有一点复杂(下文解释)。

老铁如果对快速排序的几种排序思想还有什么不明白的可以学习博主的专栏。文章最后会贴。

什么是左右指针法?一张图带你搞定:

image.gif编辑

利用递归来继续分割区间,分割后继续单趟排,最后实现整体排序,排序结束。

算法逻辑弄明白了,现在还有一个问题,那就是函数内部,不知道我们要对什么类型的数据进行排序,我们虽然使用 void * 无指定类型的指针来接收数据,内部怎么根据数据的类型来进行强制类型转换呢?不转换无法处理数据。

大家回忆一下,我们qsort()函数是不是有四个参数,其中有一个参数就是 某一个元素所占的字节大小。size。

我们是不是可以将 void * 转换成 char * ,char* 的指针每一次只能访问一个字节的内容。

举个例子:

image.gif编辑image.gif编辑

现在访问数据元素的问题解决了,那怎么交换数据元素的位置呢?

还是建立在访问元素的基础上,先找到需要交换的元素的位置,再根据 size 的大小一个字节一个字节的交换数据。

//交换数据voidSwap(char*base1, char*base2, intsize)
{
for (inti=0; i<size; ++i)//按字节转换    {
chartmp=*base1;
*base1=*base2;
*base2=tmp;
++base1;
++base2;    
    }
}

image.gif

这一趟下来,两个元素的就实现了交换。

完整版代码:

#include <stdio.h>#include <stdlib.h>intCmp_qsort(voidconst*p1, voidconst*p2)//用户输入,{
intsize1= (*(int*)p1-*(int*)p2);
returnsize1;
}
//交换数据voidSwap(char*base1, char*base2, intsize)
{
for (inti=0; i<size; ++i)//按字节转换    {
chartmp=*base1;
*base1=*base2;
*base2=tmp;
++base1;
++base2;    
    }
}
//模拟实现void_Quick_qsort(voidconst*base, intleft, intright, intsize, int(*Cmp_qsort)(voidconst*p1, voidconst*p2))
{
if (left>=right)
    {
return;
    }
intbegin=left;
intend=right;
intkey=begin;//记录坑位的下标、while (begin<end)
    {
while (begin<end&& (Cmp_qsort((char*)base+key*size, (char*)base+end*size) <=0))
--end;
while (begin<end&& (Cmp_qsort((char*)base+key*size, (char*)base+begin*size) >=0))
++begin;
Swap((char*)base+begin*size, (char*)base+end*size, size);
    }
Swap((char*)base+begin*size, (char*)base+key*size, size);
_Quick_qsort(base, left, begin-1, size, Cmp_qsort);
_Quick_qsort(base, begin+1, right, size, Cmp_qsort);
}
//过渡一下voidQuick_qsort(voidconst*base, intlen, intsize,int(*Cmp_qsort)(voidconst*p1,voidconst*p2))
{
_Quick_qsort(base, 0, len-1, size, Cmp_qsort);//左右区间写入参数,}
//打印voidPrint(int*a, intn)
{
for (inti=0; i<n; ++i)
    {
printf("%d ", a[i]);
    }
}
//快排左右指针法intmain()
{
inta[] = {6,1,2,10,7,9,9,3,4,5,10,8};
intlen=sizeof(a) /sizeof(a[0]);
intsize=sizeof(a[0]);
Quick_qsort(a, len, size,Cmp_qsort);//快速排序模拟实现Print(a, len);//打印return0;
}

image.gif

image.gif编辑

这个模拟是争对于顺序结构的数据,而链式结构要采用不同的办法。

不采用快排的挖坑发的原因是:我们需要一个类型大小的空间存坑值,这个时候我们只能根据size(一个元素所占的字节数)动态开辟一个char *的数组,一个字节一个字节的存储。如果光定义指针指向,那坑值指针,会随着指向地址的元素的变化而变化。


至此,C语言的深度解剖 qsort() 函数,博主已经分享完了,希望对大家有所帮助,如有不妥之处欢迎批评指正。

image.gif编辑

本期收录于博主的专栏——数据排序,适用于编程初学者,感兴趣的朋友们可以订阅,查看其它“经典数据排序”。排序算法_保护小周ღ的博客

感谢每一个观看本篇文章的朋友,更多精彩敬请期待:保护小周ღ  *★,°*:.☆( ̄▽ ̄)/$:*.°★*

文章存在借鉴,如有侵权请联系修改删除! image.gif编辑

相关文章
|
19天前
|
存储 C语言 开发者
【C语言】字符串操作函数详解
这些字符串操作函数在C语言中提供了强大的功能,帮助开发者有效地处理字符串数据。通过对每个函数的详细讲解、示例代码和表格说明,可以更好地理解如何使用这些函数进行各种字符串操作。如果在实际编程中遇到特定的字符串处理需求,可以参考这些函数和示例,灵活运用。
39 10
|
19天前
|
存储 程序员 C语言
【C语言】文件操作函数详解
C语言提供了一组标准库函数来处理文件操作,这些函数定义在 `<stdio.h>` 头文件中。文件操作包括文件的打开、读写、关闭以及文件属性的查询等。以下是常用文件操作函数的详细讲解,包括函数原型、参数说明、返回值说明、示例代码和表格汇总。
42 9
|
19天前
|
C语言 开发者
【C语言】数学函数详解
在C语言中,数学函数是由标准库 `math.h` 提供的。使用这些函数时,需要包含 `#include <math.h>` 头文件。以下是一些常用的数学函数的详细讲解,包括函数原型、参数说明、返回值说明以及示例代码和表格汇总。
40 6
|
19天前
|
存储 C语言
【C语言】输入/输出函数详解
在C语言中,输入/输出操作是通过标准库函数来实现的。这些函数分为两类:标准输入输出函数和文件输入输出函数。
107 6
|
19天前
|
存储 缓存 算法
【C语言】内存管理函数详细讲解
在C语言编程中,内存管理是至关重要的。动态内存分配函数允许程序在运行时请求和释放内存,这对于处理不确定大小的数据结构至关重要。以下是C语言内存管理函数的详细讲解,包括每个函数的功能、标准格式、示例代码、代码解释及其输出。
48 6
|
人工智能 C语言 vr&ar
C语言及程序设计提高例程-21 首尝排序——冒个泡
贺老师教学链接  C语言及程序设计提高 本课讲解 冒泡排序源程序 #include &lt;stdio.h&gt; #define n 10 int main( ) { int i,j,t,a[10]= {3,5,9,11,33,6,-9,-76,100,123}; for (i=0; i&lt;n-1; i++) for(j=0; j&lt;=n-i-1;
1098 0
|
2月前
|
C语言 C++
C语言 之 内存函数
C语言 之 内存函数
40 3
|
19天前
|
存储 Unix Serverless
【C语言】常用函数汇总表
本文总结了C语言中常用的函数,涵盖输入/输出、字符串操作、内存管理、数学运算、时间处理、文件操作及布尔类型等多个方面。每类函数均以表格形式列出其功能和使用示例,便于快速查阅和学习。通过综合示例代码,展示了这些函数的实际应用,帮助读者更好地理解和掌握C语言的基本功能和标准库函数的使用方法。感谢阅读,希望对你有所帮助!
31 8
|
19天前
|
C语言 开发者
【C语言】断言函数 -《深入解析C语言调试利器 !》
断言(assert)是一种调试工具,用于在程序运行时检查某些条件是否成立。如果条件不成立,断言会触发错误,并通常会终止程序的执行。断言有助于在开发和测试阶段捕捉逻辑错误。
27 5
|
1月前
|
存储 人工智能 算法
数据结构实验之C 语言的函数数组指针结构体知识
本实验旨在复习C语言中的函数、数组、指针、结构体与共用体等核心概念,并通过具体编程任务加深理解。任务包括输出100以内所有素数、逆序排列一维数组、查找二维数组中的鞍点、利用指针输出二维数组元素,以及使用结构体和共用体处理教师与学生信息。每个任务不仅强化了基本语法的应用,还涉及到了算法逻辑的设计与优化。实验结果显示,学生能够有效掌握并运用这些知识完成指定任务。
51 4