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编辑

相关文章
|
1天前
|
C语言
c语言调用的函数的声明
被调用的函数的声明: 一个函数调用另一个函数需具备的条件: 首先被调用的函数必须是已经存在的函数,即头文件中存在或已经定义过; 如果使用库函数,一般应该在本文件开头用#include命令将调用有关库函数时在所需要用到的信息“包含”到本文件中。.h文件是头文件所用的后缀。 如果使用用户自己定义的函数,而且该函数与使用它的函数在同一个文件中,一般还应该在主调函数中对被调用的函数做声明。 如果被调用的函数定义出现在主调函数之前可以不必声明。 如果已在所有函数定义之前,在函数的外部已做了函数声明,则在各个主调函数中不必多所调用的函数在做声明
16 6
|
4天前
|
搜索推荐 算法 C语言
【排序算法】八大排序(上)(c语言实现)(附源码)
本文介绍了四种常见的排序算法:冒泡排序、选择排序、插入排序和希尔排序。通过具体的代码实现和测试数据,详细解释了每种算法的工作原理和性能特点。冒泡排序通过不断交换相邻元素来排序,选择排序通过选择最小元素进行交换,插入排序通过逐步插入元素到已排序部分,而希尔排序则是插入排序的改进版,通过预排序使数据更接近有序,从而提高效率。文章最后总结了这四种算法的空间和时间复杂度,以及它们的稳定性。
31 8
|
4天前
|
搜索推荐 算法 C语言
【排序算法】八大排序(下)(c语言实现)(附源码)
本文继续学习并实现了八大排序算法中的后四种:堆排序、快速排序、归并排序和计数排序。详细介绍了每种排序算法的原理、步骤和代码实现,并通过测试数据展示了它们的性能表现。堆排序利用堆的特性进行排序,快速排序通过递归和多种划分方法实现高效排序,归并排序通过分治法将问题分解后再合并,计数排序则通过统计每个元素的出现次数实现非比较排序。最后,文章还对比了这些排序算法在处理一百万个整形数据时的运行时间,帮助读者了解不同算法的优劣。
21 7
|
14天前
|
存储 算法 程序员
C语言:库函数
C语言的库函数是预定义的函数,用于执行常见的编程任务,如输入输出、字符串处理、数学运算等。使用库函数可以简化编程工作,提高开发效率。C标准库提供了丰富的函数,满足各种需求。
|
17天前
|
存储 C语言
【c语言】字符串函数和内存函数
本文介绍了C语言中常用的字符串函数和内存函数,包括`strlen`、`strcpy`、`strcat`、`strcmp`、`strstr`、`strncpy`、`strncat`、`strncmp`、`strtok`、`memcpy`、`memmove`和`memset`等函数的使用方法及模拟实现。文章详细讲解了每个函数的功能、参数、返回值,并提供了具体的代码示例,帮助读者更好地理解和掌握这些函数的应用。
16 0
|
17天前
|
C语言
【c语言】qsort函数及泛型冒泡排序的模拟实现
本文介绍了C语言中的`qsort`函数及其背后的回调函数概念。`qsort`函数用于对任意类型的数据进行排序,其核心在于通过函数指针调用用户自定义的比较函数。文章还详细讲解了如何实现一个泛型冒泡排序,包括比较函数、交换函数和排序函数的编写,并展示了完整的代码示例。最后,通过实际运行验证了排序的正确性,展示了泛型编程的优势。
17 0
|
30天前
|
C语言 C++
C语言 之 内存函数
C语言 之 内存函数
32 3
|
21天前
|
存储 缓存 C语言
【c语言】简单的算术操作符、输入输出函数
本文介绍了C语言中的算术操作符、赋值操作符、单目操作符以及输入输出函数 `printf` 和 `scanf` 的基本用法。算术操作符包括加、减、乘、除和求余,其中除法和求余运算有特殊规则。赋值操作符用于给变量赋值,并支持复合赋值。单目操作符包括自增自减、正负号和强制类型转换。输入输出函数 `printf` 和 `scanf` 用于格式化输入和输出,支持多种占位符和格式控制。通过示例代码详细解释了这些操作符和函数的使用方法。
33 10
|
20天前
|
机器学习/深度学习 C语言
【c语言】一篇文章搞懂函数递归
本文详细介绍了函数递归的概念、思想及其限制条件,并通过求阶乘、打印整数每一位和求斐波那契数等实例,展示了递归的应用。递归的核心在于将大问题分解为小问题,但需注意递归可能导致效率低下和栈溢出的问题。文章最后总结了递归的优缺点,提醒读者在实际编程中合理使用递归。
49 7