qsort函数详解
- 前言:
- 一、qsort函数的含义
- 1.1 函数的参数
- 1.2 参数的含义
- 二、用不同类型数据,测试sqort
- 2.1 对数组内整数进行排序
- 2.2对数组内浮点数进行排序
- 2.3对字符串进行排序
- 2.4对结构体进行排序
- 三、模拟实现qsort函数
前言:
qsort()函数(quick sort)是八大排序算法中的快速排序,能够排序任意数据类型的数组其中包括整形,浮点型,字符串甚至还有自定义的结构体类型。
一、qsort函数的含义
点击网站:https://cplusplus.com/reference/——>再点击查看老版本函数超链接口(查看函数的含义)
老版本
在搜索框输入要查询的函数
查看qsort函数的含义
翻译:
1.1 函数的参数
//void qsort(void* base,//指向了需要排序的数组的第一个元素
// siez_t num, //排序的元素个数
// size_t size, //一个元素类型的大小,单位是字节。
// int (*compar)(const void*, const void*))//函数指针类型——这个函数指针指向的函数能够比较base指向数组中的两个元素。
函数有四个参数:第1个参数是个指针;size_t就是无符号整型(第2,第3个参数);第4个参数是函数指针类型。
排序由base指向的num个元素,每个元素由size个字节的长度,使用这个comper指向的函数去探测顺序(比较顺序)
1.2 参数的含义
base:base指向了要被排序的数组的第一个元素,被转换为void*。
num:base指向的数组空间元素的个数,它是一个无符号整型。
size:每个元素的单位是多少个字节的个数,它是一个无符号整型。
comper:指针指向比较两个元素的函数
int comper(const void* p1 ,const void* p2);
比较两个函数,p1指向的值和p2指向的值进行比较:
p1指向的值大于p2指向的值,就返回>0的数;
p1指向的值等于p2指向的值,就返回=0的数;
p1指向的值小于p2指向的值,就返回<0的数;
二、用不同类型数据,测试sqort
2.1 对数组内整数进行排序
整数的升序:
#include <stdio.h>
#include <stdlib.h>
int comper(const void* p1, const void* p2)
{
return (*(int*)p1 - *(int*)p2);
}
int main()
{
int arr1[10] = { 3,6,8,2,6,9,1,5,8,4 };
qsort(arr1,sizeof(arr1)/sizeof(arr1[0]),sizeof(int),comper);
int i = 0;
for (i = 0; i < sizeof(arr1) / sizeof(arr1[0]); i++)
printf("%d ", arr1[i]);
return 0;
}
2.2对数组内浮点数进行排序
#include <stdio.h>
#include <stdlib.h>
int comper(const void* p1, const void* p2)
{
return (*(float*)p1 - *(float*)p2);
}
int main()
{
float arr1[10] = { 3.33,6.66,8.88,2.22,6.66,9.99,1.11,5.55,8.88,4.44 };
qsort(arr1, sizeof(arr1) / sizeof(arr1[0]), sizeof(float), comper);
int i = 0;
for (i = 0; i < sizeof(arr1) / sizeof(arr1[0]); i++)
printf("%0.3f ", arr1[i]);
return 0;
}
2.3对字符串进行排序
#include <stdio.h>
#include <stdlib.h>
int comper(const void* p1, const void* p2)
{
return (*(char*)p1 - *(char*)p2);
}
int main()
{
char arr1[10] = "asdfghjkl";
qsort(arr1, sizeof(arr1) / sizeof(arr1[0]), sizeof(char), comper);
int i = 0;
for (i = 0; i < sizeof(arr1) / sizeof(arr1[0]); i++)
printf("%c ", arr1[i]);
return 0;
}
2.4对结构体进行排序
结构体按照字符大小进行排序
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
struct str
{
char arr[10];
int age;
};
int comper(const void* p1, const void* p2)
{
return strcmp(((struct str*)p1)->arr,((struct str*)p2)->arr);
}
int main()
{
struct str ar[3] = {{"pan",19},{"wang",21},{"li",31}};
qsort(ar, sizeof(ar) / sizeof(ar[0]), sizeof(ar[0]), comper);
int i = 0;
for (i = 0; i < sizeof(ar) / sizeof(ar[0]); i++)
printf("%s,%d\n", ar[i].arr, ar[i].age);
return 0;
}
三、模拟实现qsort函数
- 用冒泡排序的思想
- 适用于任意类型的数据排序
问题1:参数只能接受整型数组
解决1:void* 的指针
//void qsort(void* base,//指向了需要排序的数组的第一个元素
// siez_t num, //排序的元素个数
// size_t size, //一个元素类型的大小,单位是字节。
// int (*compar)(const void*, const void*))//函数指针类型——这个函数指针指向的函数能够比较base指向数组中的两个元素。
问题2:整型比较大小 < > ==
结构体比较大小,需要看成员类型在去比较。
对于不同类型的数据,不能简单的使用>去比较
解决2:将两个元素的比较方法,以函数参数的形式传递。
int comp(const void* p1, const void* p2)
{
return (*(int*)p1 - *(int*)p2);
}
总体实现:
//模拟实现qsort函数
#include <stdio.h>
int comp(const void* p1, const void* p2)
{
return (*(int*)p1 - *(int*)p2);
}
void swap(void* p1, void* p2,int si)//元素arr[j]和元素arr[j+1]进行交换,si表示交换的字节大小
{
int i = 0;
for (i = 0; i < si; i++)//把每个字节进行交换,整体就进行交换了
{
char tmp = *((char*)p1 + i);
*((char*)p1 + i) = *((char*)p2 + i);
*((char*)p2 + i) =tmp ;
}
}
void qsort(char* base, int num, int si, int(*com)(const void* ,const void*))//
{
int i, j;
for(i=0;i<num-1;i++)
for (j = 0; j < num - i - 1; j++)
{
if (com((char*)base + si * j, (char*)base + (j + 1) * si) > 0)//由小到大排序,两元素比较大小,将arr[j]和arr[j+1]的地址传递给com函数
{
swap((char*)base + si * j, (char*)base + (j + 1) * si,si);//进行交换
}
}
}
int main()
{
int arr[15] = { 1,3,5,7,9,11,13,15,14,12,10,8,6,4,2 };//定义一个整型数组
int zs = sizeof(arr) / sizeof(arr[0]);
qsort(arr, zs, sizeof(int), comp);
int i;
for (i = 0; i < zs; i++)
printf("%d ", arr[i]);
return 0;
}