c语言-qsort函数

目录

一、函数介绍

二、qsort函数的使用

1、对int类型数组排序

2、对char类型排序

3、对浮点型排序

4.比较字符串

4.1按首字母排序

4.2按长度排序

4.3按字典顺序

5.结构体排序

5.1 多级排序

三、模拟实现qsort函数

【冒泡排序的实现】

【主函数部分】

【代码详解】

 【代码实现之整型排序】

【代码实现之结构体类型排序】 

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


一、函数介绍

qsort就是C语言中的快排函数,包含在stdlib.h头文件中,函数一共有四个参数,没有返回值。

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

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

对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 函数的返回值也是整型数据(这个是根据对应的场景,选择比较方式),即可得到相应的结果。

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

对于陌生的库函数,可通过cplusplus网站来了解它

【qsort参数介绍】

【比较函数的返回值】

qsort函数大致的模板为

int compare(const void* p1, const void* p2) //
{
	return p1 - p2; //返回的是升序
	return p2 - p1; //返回的是降序   
    
    //注:p1和p2的类型根据实际情况写
}
 
int main()
{
	int arr[] = { 1,3,4,6,7,2,10,8,5,9 };
	int sz = sizeof(arr) / sizeof(arr[0]);//计算数组元素个数 40 / 4 = 10
 
	qsort(arr, sz, sizeof(arr[0]), compare);
	//arr - 指向要排序的数组的第一个元素的指针。
	//sz -  由 arr 指向的数组中元素的个数
	//sizeof(arr[0]) - 数组中每个元素的大小,以字节为单位。
	//compar - 用来比较两个元素的函数。
 
	return 0;
}

Q:为什么compare的形参的两个参数是void*类型?

因为qsort函数可以实现对数组(int)、字符串(char)、结构体(stuct)等类型进行升序或降序排序,void*是无具体类型的指针,是不介意类型的,就像一个“垃圾桶”,任意的类型的地址都能往void*塞,但就是不能对其直接使用(解引用操作,++,--等等)。若想使用,只要进行强制类型转化即可。

二、qsort函数的使用

1、对int类型数组排序

//升序的情况
#include   //使用qsort需要包含头文件
#include   
int compare_int(const void* p1, const void* p2) 
{
	return *(int*)p1 - *(int*)p2; //强制类型转化并解引用
}
 
int main()
{
	int arr[] = { 1,3,4,6,7,2,10,8,5,9 };
	int sz = sizeof(arr) / sizeof(arr[0]);
 
	qsort(arr, sz, sizeof(arr[0]), compare_int);
	
	for (int i = 0; i < sz; i++)
	{
		printf("%d ", arr[i]);
	}
 
	return 0;
}
//降序的情况
#include   //使用qsort需要包含头文件
#include   
int compare_int(const void* p1, const void* p2) 
{
	return *(int*)p2 - *(int*)p1; //强制类型转化并解引用
}
 
int main()
{
	int arr[] = { 1,3,4,6,7,2,10,8,5,9 };
	int sz = sizeof(arr) / sizeof(arr[0]);
 
	qsort(arr, sz, sizeof(arr[0]), compare_int);
	
	for (int i = 0; i < sz; i++)
	{
		printf("%d ", arr[i]);
	}
 
	return 0;
}

2、对char类型排序

//升序的情况
#include 
#include 
 
int compare_char(const void* p1, const void* p2) 
{
	return *(char*)p1 - *(char*)p2; //强制类型转化并解引用
}
 
int main()
{
	char arr[] = { 'f', 'b','e','a','d','c'};
	int sz = sizeof(arr) / sizeof(arr[0]);
 
	
	qsort(arr, sz, sizeof(arr[0]), compare_char);
	
	for (int i = 0; i < sz; i++)
	{
		printf("%c ", arr[i]);
	}
	return 0;
}

3、对浮点型排序

#include 
#include 
 
int compare_double(const void* p1, const void* p2) 
{
	return (*(double*)p1 > *(double*)p2 ? 1 : -1); //三目操作符
}
 
int main()
{
	double arr[] = { 3.14,2.6,2.3,1.7};
	int sz = sizeof(arr) / sizeof(arr[0]);
 
	
	qsort(arr, sz, sizeof(arr[0]), compare_double);
	
	for (int i = 0; i < sz; i++)
	{
		printf("%lf ", arr[i]);
	}
 
 
	return 0;
}

注意

用qsort对浮点型的一定要用三目运算符,由于浮点数的精度问题,如果是两个很接近的数相减,则可能返回一个接近0的小数,而compare_double的返回值是int型,因此会将这个小数返回0。

4.比较字符串

4.1按首字母排序

#include
#include
#define L 10
#define K 10
int inc(const void *a, const void *b)
{
	return *(char *)a - *(char *)b;
 } 
int main ()
{
	char a[L][K] = {
		"rbsc",
		"jcse",
		"efgd",
		"arbs",
		"bbs",
		"cbfe",
		"dgafg" ,
		"ewqrta",
		"ofgd",
		"mbcv",
	};
	qsort(a, L, sizeof(char) * K, inc);
	for (int i = 0; i < L; i++)
	{
		printf("%s\n", a[i]);
	}
 } 

4.2按长度排序

#include
#include
#include
#define L 10
#define K 10
int inc(const void *a, const void *b)
{
	return strlen((char *)a) > strlen((char *)b) ? 1 : -1;
 } 
int main ()
{
	char a[L][K] = {
		"rbsc",
		"jcsse",
		"efgdsd",
		"arbs",
		"bbs",
		"cbfefaa",
		"dgafg" ,
		"ewqrta",
		"ofgd",
		"mbcv312",
	};
	qsort(a, L, sizeof(char) * K, inc);
	for (int i = 0; i < L; i++)
	{
		printf("%s\n", a[i]);
	}
 } 

4.3按字典顺序

#include
#include
#include
#define L 10
#define K 10
int inc(const void *a, const void *b)
{
	return strcmp((char * )a, (char *)b);
 } 
int main ()
{
	char a[L][K] = {
		"rbsc",
		"jcsse",
		"afgdsd",
		"arbs",
		"abs",
		"cbfefaa",
		"cgafg" ,
		"ewqrta",
		"ofgd",
		"mbcv312",
	};
	qsort(a, L, sizeof(char) * K, inc);
	for (int i = 0; i < L; i++)
	{
		printf("%s\n", a[i]);
	}
 } 

5.结构体排序

5.1 多级排序

结构体体的三级排序测试:
第一级是对学生成绩整体从小到大排序;
第二级是对相同成绩的学生,按照姓名进行排序;
第三级是对相同成绩、姓名的学生,按照学号进行排序;

#include
#include
#include 
typedef struct student
{
    int id;
	char name[10];
    int grade;	
}student;
 
int cmp1(const void *a, const void *b)//一级排序
{
    student *s1 = (student*)a;
    student *s2 = (student*)b;
    return s1->id - s2->id;
}

int cmp2(const void *a,const void *b)//二级排序
{
	student *s1 = (student*)a;
    student *s2 = (student*)b;
	if(strcmp(s1->name , s2->name) != 0)
		return strcmp(s1->name , s2->name);	
	else	
		return s1->id - s2->id;				
}

int cmp3(const void *a,const void *b)//三级排序
{
	student *s1 = (student*)a;
    student *s2 = (student*)b;
	if(s1->grade != s2->grade)	
		return s1->grade - s2->grade;
	else
	{
		if(strcmp(s1->name , s2->name) != 0)
			return strcmp(s1->name , s2->name);
		else
			return s1->id - s1->id;
	}
}

int main()
{
    int i,N,C;
	scanf("%d %d",&N,&C);
	
	student *stu;
	stu=(student*)malloc(N*sizeof(student));

	for(i = 0 ; i < N ; i++)
		scanf("%d %s %d" , &stu[i].id , stu[i].name , &stu[i].grade);
	switch(C)
	{
		case 1:	qsort(stu, N, sizeof(student), cmp1);break;//一级排序
		case 2:	qsort(stu, N, sizeof(student), cmp2);break;//二级排序
		case 3:	qsort(stu, N, sizeof(student), cmp3);break;//三级排序
	}
	printf("排序结果:\n");
	for(i = 0 ; i < N ; i++)
		printf("%03d %s %d\n" , stu[i].id , stu[i].name , stu[i].grade);
    return 0;
}

三、模拟实现qsort函数

模拟实现qsort函数是要基于冒泡排序实现的    (冒泡排序讲解)

【冒泡排序的实现】

#include 
void Sort(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])
			{
				int tmp = arr[j];
				arr[j] = arr[j + 1];
				arr[j + 1] = tmp;
			}
		}
	}
}
int main()
{
	int arr[] = { 2,6,8,7,6,0,1,5,9,3 };
	int sz = sizeof(arr) / sizeof(arr[0]);
	Sort(arr, sz);
	for (int i = 0; i < sz; i++)
	{
		printf("%d ", arr[i]);
	}
	return 0;
}

现要求改造冒泡排序,使整个函数可以排序任意类型的数组

以整型数组为例

【主函数部分】

#include 
int cmp_int(const void* p1, const void* p2)
{
	return *(int*)p1 - *(int*)p2;
}
 
int main()
{ 
	int arr[] = { 2,6,8,7,6,0,1,5,9,3 };
	int sz = sizeof(arr) / sizeof(arr[0]);
 
	Sort(arr, sz, sizeof(arr[0]), cmp_int); 
 
	for (int i = 0; i < sz; i++)
	{
		printf("%d ", arr[i]);
	}
	return 0;
}

【Sort函数部分】

参考qsort函数的参数

void Swap(char* x1, char* x2, int width)
{
	//因为不知道是什么类型,要一个字节一个字节交换
	for (int i = 0; i < width; i++)
	{
		char tmp = *x1;
		*x1 = *x2;
		*x2 = tmp;
		x1++;
		x2++;
	}
}
 
//void* base - 要求排序不同类型的数组,void*恰好能接收任意类型
//int sz - 元素个数
//int width - 一个元素的大小
//int (*p)(const void*, const void*)  函数传参函数指针接收
//size_t - 无符号整型
 
void Sort(void* base, size_t sz, size_t width, int (*p)(const void*, const void*))
{
	for (size_t i = 0; i < sz - 1; i++)
	{
		for (size_t j = 0; j < sz - 1 - i; j++)
		{
            //通过函数指针p调用的函数cmp_int,所以这是个回调函数
			if (p((char*)base + j * width,(char*)base + (j + 1) * width) > 0)
			{
				//写一个Swap函数来交换
				Swap((char*)base + j * width, (char*)base + (j + 1) * width, width);
			}
		}
	}
}

 

【代码详解】

 【代码实现之整型排序】

 

int cmp_int(const void* p1, const void* p2)
{
	return *(int*)p1 - *(int*)p2;
}
 
void Swap(char* x1, char* x2, int width)
{
	for (int i = 0; i < width; i++)
	{
		char tmp = *x1;
		*x1 = *x2;
		*x2 = tmp;
		x1++;
		x2++;
	}
}
 
void Sort(void* base, size_t sz, size_t width, int (*p)(const void*, const void*))
{
	for (size_t i = 0; i < sz - 1; i++)
	{
		for (size_t j = 0; j < sz - 1 - i; j++)
		{
			if (p((char*)base + j * width,(char*)base + (j + 1) * width) > 0)
			{
				Swap((char*)base + j * width, (char*)base + (j + 1) * width, width);
			}
		}
	}
}
 
int main()
{ 
	int arr[] = { 2,6,8,7,6,0,1,5,9,3 };
	int sz = sizeof(arr) / sizeof(arr[0]);
 
	Sort(arr, sz, sizeof(arr[0]), cmp_int); 
 
	for (int i = 0; i < sz; i++)
	{
		printf("%d ", arr[i]);
	}
	return 0;
}

【代码实现之结构体类型排序】 

以排列姓名为例

struct Stu
{
	char name[20];
	int age;
};
int compare_name(const void* p1, const void* p2)
{
	return strcmp(((struct Stu*)p1)->name, ((struct Stu*)p2)->name);
}
 
 
void Swap(char* x1, char* x2, int width)
{
	for (int i = 0; i < width; i++)
	{
		char tmp = *x1;
		*x1 = *x2;
		*x2 = tmp;
		x1++;
		x2++;
	}
}
 
void Sort(void* base, size_t sz, size_t width, int (*p)(const void*, const void*))
{
	for (size_t i = 0; i < sz - 1; i++)
	{
		for (size_t j = 0; j < sz - 1 - i; j++)
		{
			if (p((char*)base + j * width,(char*)base + (j + 1) * width) > 0)
			{
				Swap((char*)base + j * width, (char*)base + (j + 1) * width, width);
			}
		}
	}
}
 
int main()
{ 
	struct Stu s[3] = { {"张三",10},{"李四",30},{"王五",21} };
	int sz = sizeof(s) / sizeof(s[0]);
 
	Sort(s, sz, sizeof(s[0]), compare_name);
 
	for (int i = 0; i < sz; i++)
	{
		printf("%s %d\n", s[i].name,s[i].age);
	}
	return 0;
}

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

#include 
#include 
 
int Cmp_qsort(void const* p1, void const* p2)//用户输入,
{
	int size1 = (*(int*)p1 - *(int*)p2);
	return size1;
}
 
//交换数据
void Swap(char* base1, char* base2, int size)
{
	for (int i = 0; i < size; ++i)//按字节转换
	{
		char tmp = *base1;
		*base1 = *base2;
		*base2 = tmp;
		++base1;
		++base2;	
	}
}
 
//模拟实现
void _Quick_qsort(void const* base, int left, int right, int size, int(*Cmp_qsort)(void const* p1, void const* p2))
{
	if (left >= right)
	{
		return;
	}
 
	int begin = left;
	int end = right;
	int key = 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);
 
}
 
 
//过渡一下
void Quick_qsort(void const* base, int len, int size,int(*Cmp_qsort)(void const *p1,void const *p2))
{
	_Quick_qsort(base, 0, len - 1, size, Cmp_qsort);//左右区间写入参数,
}
 
//打印
void Print(int* a, int n)
{
	for (int i = 0; i < n; ++i)
	{
		printf("%d ", a[i]);
	}
}
 
//快排左右指针法
int main()
{
	int a[] = {6,1,2,10,7,9,9,3,4,5,10,8};
	int len = sizeof(a) / sizeof(a[0]);
	int size = sizeof(a[0]);
 
	Quick_qsort(a, len, size,Cmp_qsort);//快速排序模拟实现
 
	Print(a, len);//打印
	return 0;
 
}

文章存在借鉴,如有侵权请联系修改删除!

你可能感兴趣的:(C语言,c语言,开发语言)