【冒泡排序】模仿qsort的功能实现一个通用的冒泡排序


前言

上一章讲了qsort如何排序各种类型数据,本章继续学习如何模仿qsort的功能实现一个通用的冒泡排序


这个冒泡排序要具备:
1.使用冒泡排序的思想
2.适应于任意类型数据的排序

void* 是实现一个通用的冒泡排序最核心的部位
因为void* 类型的指针可以接收任意类型的地址

                        假设排序整型

曾经学的冒泡排序存在着一些局限性

在这里插入图片描述
解决方法是:
在这里插入图片描述
对于问题三,后面会讲到

首先第一步:写一个main()函数,分装一个test1函数

int main()
{
    
    
	test2();
	return 0;
}

test1将会模仿qsort的功能:
在这里插入图片描述
四个函数的意思是:
void qsort(
void* base,//指向需要排序的数组的第一个元素
size_t num,//排序的元素个数
size_t size,//一个元素的大小,单位是字节
int(*cmp)(const void*, const void*)
);//函数指针类型-这个函数指针指向的函数,能够比较base指向数组中的两个元素

test1函数 用来描写类型的性状

test1函数里的内容具有随变性,不是固定的,但是有一个固定的框架
这里面就写你要对什么数据类型进行冒泡排序。包含四个点:

①写清楚是什么数组,有什么元素。
②一个元素的大小。
③写好等会要调用的bubble_sor函数,包括它的四个参数。
④最后分装一个打印函数,打印结果的时候要调用它。

//代码如下:
void test1()
{
    
    
	int arr[] = {
    
     2,1,4,6,5,3,8,0,9,7 };//整型数组,有10个数字
	int sz = sizeof(arr) / sizeof(arr[0]);//一个元素大小
	bubble_sort(arr, sz, sizeof(arr[0]), cmp_int);//模仿qsort函数的参数
	print(arr, sz);//打印函数
}

在test1创建了bubble_int 函数,下一步就是实现它,分两步走

步骤一:写函数参数

bubble_int函数的参数,如何才能接收任意类型数据呢?
图二解决一详细说了解决方法:模仿qsort函数参数

另外在bubble_int函数里,大部分代码都是固定的,可以说是一个模板,不能随意改动,而每种类型的比较方法都是不同的,所以比较的方法就不能写在bubble_int函数里。
需要另外写一个交换函数。在bubble_int函数里调用以下就可以了。
这就是为什么qsort函数的第四个参数是cmp_int函数了。所以bubble_int的第四个参数是cmp_int函数指针。

bubble_int参数总结概括为 传某个类型数组的起始位置+数组个数+一个元素大小+比较方法的函数指针
所以bubble_int的参数如下:


void bubble_sort(void* base, int num, int size, int(*cmp)(const void*, const void*))
{
    
    
	
}

步骤二:写具体代码实现比较

bubble_int里面的内容简单概括为:循环对比两个数
上图指出:不管是整形数据,浮点型数组,还是结构体数据,它们的趟数是不会变的,一趟内部比较的对数也不会变。
所以这个趟数循环的代码也是固定的。

代码如下:

void bubble_sort(void* base, int num, int size, int(*cmp)(const void*, const void*))
{
    
    
	int i = 0;
	//趟数
	for (i = 0; i < num - 1; i++)
	{
    
    
		int j = 0;
		//一趟内部比较的对数 
		for (j = 0; j < num - 1 - i; j++)
		{
    
    
			if()两个元素比较
		}
	}
}

图二说到,在if语句里的问题是:不能简单的让两个数用>来比较。
解决方法是要调用cmp_int函数指针,

所以我们先完善一下cmp_int函数。

cmp_int的两个参数是const修饰的void*指针,两个元素分别命名为p1,p2
比较方法是:两个数作差,就是p1-p2
因为已经知道是整型数组,所以要把p1,p2强制类型转换成整型,再解引用找到p1和p2的值。
注意:记得写return,因为要把作差结果返回给cmp,cmp_int前面也要用int修饰,因为有返回值

int cmp_int(const void*p1, const void*p2)
{
    
    
	return (*(int*)p1 - *(int*)p2);
}

接下来继续完善if()里面的内容,

后面的步骤如下:

1.调用cmp函数

2.cmp函数里传两个元素的地址

因为数组类型不同,导致元素个数不同,每个元素的大小也不同,有的是一个字节,有的是四个字节,所以两个元素的地址是不一样的。
要实现通用的冒泡排序,要让cmp函数拿到不同类型的元素,就要传它们的地址过去。
所以有一个通用的地址计算公式,能找到任意数组类型的两个元素的地址。
第一个元素地址(char*)base+j*size,
第二个元素地址(char *)base+(j+1) * size

j表示数组下标,size表示字节大小,(char * )base表示是第一个元素地址,单位是一个字节
解释:
在这里插入图片描述
为什么是char类型的指针,因为一个一个字节找更准确
如下图所示:
在这里插入图片描述
当cmp接收地址,并把作差结果返回来时,如果结果小于0,就不用两数交换,如果结果大于0才需要交换位置,所以大于0才能进入if语句。

void bubble_sort(void* base, int num, int size, int(*cmp)(const void*, const void*))
{
    
    
	int i = 0;
	for (i = 0; i < num - 1; i++)
	{
    
    
		int j = 0;
		for (j = 0; j < num - 1 - i; j++)
		{
    
    
			if (cmp( (char*)base + j * size, (char*)base + (j + 1) * size) > 0)
			{
    
    
				
			}
		}
	}
}

3.cmp_int的参数用指针接收两个元素的地址

4.cmp_int对两个元素解引用,作差

5.cmp_int返回结果给cmp

6.cmp判断结果是否>0

7.>0,进入循环

8.循环里调用一个交换函数Swap,传两个元素的地址,和大小

调用Swap传参时要把两个元素的地址传过去,还要传两个元素的大小siez,因为不传size,Swap函数不知道要交换多少个字节。

void bubble_sort(void* base, int num, int size, int(*cmp)(const void*, const void*))
{
    
    
	int i = 0;
	for (i = 0; i < num - 1; i++)
	{
    
    
		int j = 0;
		for (j = 0; j < num - 1 - i; j++)
		{
    
    
			if (cmp( (char*)base + j * size, (char*)base + (j + 1) * size) > 0)
			{
    
    
				Swap(cmp((char*)base + j * size, (char*)base + (j + 1) * size),size);
			}
		}
	}
}

9.Swap函数接收到buf1和buf2的地址,以及大小

buf1作为地址要用指针接收,而且是char*指针(第2点讲到原因)
一个一个字节交换的
比如要交换5和4,5的地址是05 00 00 00,4的地址是04 00 00 00,
Swap接收到地址和大小,便开始一个字节一个字节交换。
在这里插入图片描述
代码如下

void Swap(char* buf1,char* buf2,int size)
{
    
    
	char tmp = 0;
	int i = 0;
	for (i = 0; i < size; i++)
	{
    
    
		tmp = *buf1;//交换元素,不是交换地址
		*buf1 = *buf2;
		*buf2 = tmp;
		buf1++;//地址往后找一个字节
		buf2++;
	}
}

10.分装打印函数,把结果打印出来

记得函数参数用指针接收,sz本身是地址就不用指针接收

void print(int* arr, int sz)
{
    
    
	int i = 0;
	for (i = 0; i < sz; i++)
	{
    
    
		printf("%d ", arr[i]);
	}
}

总结

完整的代码如下:

#include<stdio.h>

void Swap(char* buf1,char* buf2,int size)
{
    
    
	char tmp = 0;
	int i = 0;
	for (i = 0; i < size; i++)
	{
    
    
		tmp = *buf1;
		*buf1 = *buf2;
		*buf2 = tmp;
		buf1++;
		buf2++;
	}
}

void print(int* arr, int sz)
{
    
    
	int i = 0;
	for (i = 0; i < sz; i++)
	{
    
    
		printf("%d ", arr[i]);
	}
}

int cmp_int(const void*p1, const void*p2)
{
    
    
	return (*(int*)p1 - *(int*)p2);
}

void bubble_sort(void* base, int num, int size, int(*cmp)(const void*, const void*))
{
    
    
	int i = 0;
	for (i = 0; i < num - 1; i++)
	{
    
    
		int j = 0;
		for (j = 0; j < num - 1 - i; j++)
		{
    
    
			if (cmp( (char*)base + j * size, (char*)base + (j + 1) * size) > 0)
			{
    
    
				Swap((char*)base + j * size, (char*)base + (j + 1) * size,size);
			}
		}
	}
}

void test1()
{
    
    
	int arr[] = {
    
     3,5,2,0,7,9,4,1,6,8 };
	int sz = sizeof(arr) / sizeof(arr[0]);
	bubble_sort(arr, sz, sizeof(arr[0]), cmp_int);
	print(arr,sz);
}
int main()
{
    
    
	test1();
	return 0;
}

以上就是模仿qsort的功能实现一个通用的冒泡排序的方法,希望对您有帮助,感谢关注感谢点赞感谢收藏!

猜你喜欢

转载自blog.csdn.net/2301_76496134/article/details/131839253