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

文章目录


前言

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


这个冒泡排序要具备:

1.使用冒泡排序的思想

2.适应于任意类型数据的排序

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

复制代码
                        假设排序整型

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

解决方法是:

对于问题三,后面会讲到

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

cpp 复制代码
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函数,包括它的四个参数。

④最后分装一个打印函数,打印结果的时候要调用它。

cpp 复制代码
//代码如下:
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的参数如下:

cpp 复制代码
void bubble_sort(void* base, int num, int size, int(*cmp)(const void*, const void*))
{
	
}

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

bubble_int里面的内容简单概括为:循环对比两个数

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

代码如下:

cpp 复制代码
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修饰,因为有返回值

cpp 复制代码
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语句。

cpp 复制代码
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函数不知道要交换多少个字节。

cpp 复制代码
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接收到地址和大小,便开始一个字节一个字节交换。

代码如下

cpp 复制代码
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本身是地址就不用指针接收

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

总结

完整的代码如下:

cpp 复制代码
#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的功能实现一个通用的冒泡排序的方法,希望对您有帮助,感谢关注感谢点赞感谢收藏!

相关推荐
虾球xz1 小时前
CppCon 2018 学习:EFFECTIVE REPLACEMENT OF DYNAMIC POLYMORPHISM WITH std::variant
开发语言·c++·学习
Chef_Chen1 小时前
从0开始学习R语言--Day38--辛普森多样性指数
学习
Allen_LVyingbo1 小时前
Python常用医疗AI库以及案例解析(2025年版、上)
开发语言·人工智能·python·学习·健康医疗
DKPT1 小时前
Java组合模式实现方式与测试方法
java·笔记·学习·设计模式·组合模式
津津有味道2 小时前
Qt C++串口SerialPort通讯发送指令读写NFC M1卡
linux·c++·qt·串口通信·serial·m1·nfc
让我们一起加油好吗3 小时前
【C++】list 简介与模拟实现(详解)
开发语言·c++·visualstudio·stl·list
傅里叶的耶3 小时前
C++系列(二):告别低效循环!选择、循环、跳转原理与优化实战全解析
c++·visual studio
猫猫的小茶馆3 小时前
【STM32】预分频因子(Prescaler)和重装载值(Reload Value)
c语言·stm32·单片机·嵌入式硬件·mcu·51单片机
JeffersonZU3 小时前
Linux/Unix文件IO(文件描述符、原子操作、文件数据结构、open、read、write、fcntl、dup)
linux·c语言·unix·gnu
Vitta_U3 小时前
MFC的List Control自适应主界面大小
c++·list·mfc