模仿qsort实现一个通用的冒泡排序

目录

前言

模仿

排序整型数组

排序结构体数组

排序字符数组


前言

qsort在前面我们讲到底层逻辑是快速排序的方式,是不是可以发现有了qsort来进行排序的话,就更加的方便快捷,我们在使用的时候

一方面,代码量会大大的减少

另一方面,可以排序任意类型的数据

是不是发现了这函数的魅力呢?

那么今天我们也来当一个小小开发员来模仿qsort的功能实现一个通用的冒泡排序

我们先来简单的回顾一下冒泡排序:

#include<stdio.h>
void print_arr(int* arr, int sz)
{for (int i = 0; i < sz; i++)printf("%d ", *(arr + i));
}
void bubble_sort(int* arr, int sz)
{int i = 0;for (i = 0; i < sz - 1; i++)//趟数{int j = 0;for (j = 0; j < sz - 1 - i; j++)//每次比较的次数少一次{if (*(arr+j)>*(arr+j+1)){int temp = *(arr + j);*(arr + j) = *(arr + j + 1);*(arr + j + 1) = temp;}}}
}
int main()
{int arr[10] = { 2,4,1,6,9,5,3,8,0,7 };int sz = sizeof(arr) / sizeof(arr[0]);//数组元素个数szprintf("排序前:\n");print_arr(arr, sz);bubble_sort(arr, sz);printf("\n排序前:\n");print_arr(arr, sz);return 0;
}

模仿

目前我们看来,冒泡排序似乎只可以排序整型,那么我们可不可以把它模拟像qsort那样可以排序任意类型的数据呢?

我们一起来试试,既然是模仿qsort那么和qsort应该是一样的函数参数

void imitate_bubble_sort(void* base, int count,int wideth,int (*cmp)(void*,void*))
//  base   ——排序数组的首元素地址
//  count  ——排序数组的元素个数
//  wideth ——排序数组一个元素的字节长度
//  cmp    ——函数指针(具体比较的函数的地址)

不同类型的数据在imitate_bubble_sort函数内部只是比较大小的代码不一样,所以就需要自己写比较的函数。

在比较的时候,因为不同类型的字节数不一样,我们可以根据传过去的width也就是排序数组一个元素的字节长度来访问不同大小的空间进而进行比较,如果满足条件再一个字节一个字节的进行交换,因为正好char类型是一个字节的,我们就可以把void*强制类型转换为char*

排序整型数组

具体代码如下:

#include<stdio.h>
void print_arr(int* arr, int sz)
{for (int i = 0; i < sz; i++)printf("%d ", *(arr + i));
}
int cmp_int(void* p1, void* p2)
{return *((int*)p1) - *((int*)p2);//根据具体的数据类型来写
}
void swap(char* p1, char* p2,int width)
{for (int i = 0; i < width; i++){char temp = *(p1 + i);*(p1 + i) = *(p2 + i);*(p2 + i) = temp;}//一个字节一个字节进行交换
}
void imitate_bubble_sort(void* base, int count,int width,int (*cmp)(void*,void*))
//  base   ——排序数组的首元素地址
//  count  ——排序数组的元素个数
//  wideth ——排序数组一个元素的字节长度
//  cmp    ——函数指针(具体比较的函数的地址)
{int i = 0;for (i = 0; i < count - 1; i++)//趟数{int j = 0;for (j = 0; j < count - 1 - i; j++){if (cmp((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[10] = { 2,4,1,6,9,5,3,8,0,7 };int sz = sizeof(arr) / sizeof(arr[0]);//数组元素个数szprintf("排序前:\n");print_arr(arr, sz);imitate_bubble_sort(arr, sz, sizeof(int), cmp_int);printf("\n排序后:\n");print_arr(arr, sz);return 0;
}

这里我们可以看到,它成功排序了整型,那么其它类型的数据可不可以呢?

排序结构体数组

我们来试试结构体类型的:

#include<stdio.h>
#include<string.h>//strcmp头文件
struct Student
{char name[20];int age;
};
void print_arr(struct Student* arr, int sz)
{for (int i = 0; i < sz; i++)printf("%s %d\n", (arr + i)->name, (arr + i)->age);
}
int cmp_student_by_name(void* p1, void* p2)
{return strcmp(((struct Student*)p1)->name, ((struct Student*)p2)->name);
}
void swap(char* p1, char* p2, int width)
{for (int i = 0; i < width; i++){char temp = *(p1 + i);*(p1 + i) = *(p2 + i);*(p2 + i) = temp;}
}
void imitate_bubble_sort(void* base, int count, int width, int(*cmp)(void*, void*))
{int i = 0;for (i = 0; i < count - 1; i++){int j = 0;for (j = 0; j < count - 1 - i; j++){if (cmp((char*)base + j * width, (char*)base + (j + 1) * width) > 0)swap((char*)base + j * width, (char*)base + (j + 1) * width, width);}}
}
int main()
{struct Student stu_arr[3] = { {"lihua",24},{"wangdan",56},{"anna",18} };int sz = sizeof(stu_arr) / sizeof(stu_arr[0]);printf("排序前:\n");print_arr(stu_arr, sz);imitate_bubble_sort(stu_arr, sz, sizeof(stu_arr[0]), cmp_student_by_name);printf("\n排序后:\n");print_arr(stu_arr, sz);return 0;
}

排序字符数组

#include<stdio.h>
#include<string.h>
void print_arr(char* arr, int sz)
{for (int i = 0; i < sz; i++)printf("%-3c", arr[i]);
}
int cmp_char(void* p1, void* p2)
{return strcmp((char*)p1, (char*)p2);
}
void swap(char* p1, char* p2, int width)
{for (int i = 0; i < width; i++){char temp = *(p1 + i);*(p1 + i) = *(p2 + i);*(p2 + i) = temp;}
}
void imitate_bubble_sort(void* base, int count, int width, int(*cmp)(void*, void*))
{int i = 0;for (i = 0; i < count - 1; i++){int j = 0;for (j = 0; j < count - 1 - i; j++){if (cmp((char*)base + j * width, (char*)base + (j + 1) * width) > 0)swap((char*)base + j * width, (char*)base + (j + 1) * width, width);}}
}
int main()
{char c[] = "dnjffskajkb";//c——字符数组int sz = sizeof(c) / sizeof(c[0]);printf("排序前:\n");print_arr(c, sz);imitate_bubble_sort(c, sz, sizeof(c[0]), cmp_char);printf("\n排序后:\n");print_arr(c, sz);return 0;
}

可以看见无论是结构体数组,还是字符数组,我们的代码得到了我们想要的结果!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/28100.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

北京多商入驻app开发项目的主要优势及功能

多商入驻app开发项目的定义 随着电子支付技术的不断成熟&#xff0c;全国各地的消费者通过网络在线上购物的频率越来越高&#xff0c;为此&#xff0c;多商入驻app开发项目应用而生。各商家也纷纷开始申请入驻商城平台&#xff0c;开设自己的店铺。 图片来源&#xff1a;unspl…

【CT】LeetCode手撕—121. 买卖股票的最佳时机

目录 题目1- 思路2- 实现⭐121. 买卖股票的最佳时机——题解思路 2- ACM实现 题目 原题连接&#xff1a;121. 买卖股票的最佳时机 1- 思路 模式识别 模式1&#xff1a;只能某一天买入 ——> 买卖一次 ——> dp 一次的最大利润 动规五部曲 1.定义dp数组&#xff0c;确…

数据结构之线性表(2)

顺序表中的动态存储 上文我们了解到了顺序表中的静态顺序表的相关操作&#xff0c;今天我们来学习动态顺序表的知识。 为什么会存在动态顺序表呢&#xff1f;&#xff1f; 原因&#xff1a;静态顺序表给定的数据容量固定&#xff0c;多了浪费&#xff0c;少了不够用。 首先我…

【Python深度学习】——信息量|熵

【Python深度学习】——信息量|熵 假设1. 信息量1.1 含义1.2 信息量的公式: 2. 熵Entropy2. 含义2.2 熵的计算公式:2.3 熵的作用 假设 例子&#xff1a;掷硬币 假设我们有一个公平的硬币。这个硬币有两个面&#xff1a;正面&#xff08;H&#xff09;和反面&#xff08;T&…

一. 做一个前后端分离的电商项目(技术栈 : springboot+mybatis-plus+vue) 的前期准备

前期准备 ---- 项目创建和配置 一.创建springboot项目二.项目前期准备工作1. 修改springboot和jdk版本号2.Web请求处理(1) 添加web依赖(2) 测试是否能够成功访问(3) 修改端口号(4) 创建数据库 3. 连接数据库(1) 添加依赖(2)配置application.properties文件(3)添加包扫描 Mapper…

Validation校验

文章目录 Validation校验作用依赖坐标UserController接收客户端注册用户请求的方法请求参数封装实体User的结构校验分组 Validation校验 作用 服务端接收前端传递的请求从参数的时候&#xff0c;可以对请求参数进行自动校验。 场景&#xff1a;通过postman向服务端发送一个注…

《检索技术核心20讲》进阶篇之LSM树

背景 学习极客实践课程《检索技术核心20讲》https://time.geekbang.org/column/article/215243&#xff0c;文档形式记录笔记。 内容 磁盘和内存数据读取特点 工业界中数据量往往很庞大&#xff0c;比如数据无法全部加载进内存&#xff0c;无法支持索引的高效实时更新&…

视频格式转换avi格式怎么弄?分享视频转换方法

视频格式转换avi格式怎么弄&#xff1f;AVI作为一种广泛支持的视频格式&#xff0c;能够在多种设备和播放器上顺畅播放&#xff0c;确保我们的视频内容能够无障碍地分享给朋友或上传至各大平台。其次&#xff0c;AVI格式通常具有较好的兼容性&#xff0c;能够避免格式转换过程中…

修改yarn、npm、pnpm为国内镜像源

国内由于网络的原因&#xff0c;使用官方的npm、yarn、pnpm访问下载依赖库会很慢&#xff0c;有时候还会出现无法访问的情况&#xff0c;这时候就需要我们给npm、yarn、pnpm换一个国内的镜像源的&#xff0c;一般的我们可以将镜像换成淘宝的源&#xff0c;由于平时比较常用到的…

用飞书写博客,并自动部署

feishu-vitepress 用飞书写博客,并自动部署 目前的静态博客如vitepress&#xff0c;主要是用markdown来写内容。markdown虽然可读性比较好&#xff0c;但是在文章中贴图片有点麻烦&#xff0c;需要先保存图片到asset目录下&#xff0c;再在markdown中写图片地址。 平时工作主要…

优化查询性能:DolphinDB 时间类型数据比较规则详解

在数据库中&#xff0c;时间是一种常见的数据类型。在处理时间数据时&#xff0c;比较操作是非常常见的需求。然而&#xff0c;在不同的场景下&#xff0c;对时间类型数据进行比较时应用的规则不同。本文将从 DolphinDB 支持的时间类型开始&#xff0c;由浅入深分别介绍时间类型…

47.PyCharm P版突然无法启动

目录 1.启动cmd.exe&#xff0c;进到pycharm\bin目录&#xff0c;启动.\pycharm.bat&#xff0c;如果正常&#xff0c;就像下面这个样子&#xff0c;如果不正常&#xff0c;则会报错&#xff0c; 2.用记事本打开pycharm.bat文件&#xff0c;加上以下代码后 今晨&#xff0c;无…

《C++ Primer》导学系列:第 3 章 - 字符串、向量和数组

3.1 命名空间的使用 概述 命名空间是C中的一种机制&#xff0c;用于组织代码并避免名字冲突。通过使用命名空间&#xff0c;可以将全局作用域中的标识符组织到逻辑分组中&#xff0c;从而提高代码的可读性和可维护性。命名空间在大型项目中尤为重要&#xff0c;因为它们可以防…

关于C#导出Word时报错“{00020970-0000-0000-C000-000000000046}加载类型库/DLL 时出错”的解决办法

之前还运行正常的程序&#xff0c;突然发现导出Word的时候会报错&#xff0c;报错内容&#xff1a; System.InvalidCastException:“Unable to cast COM object of type ‘Microsoft.Office.Interop.Word.ApplicationClass’ to interface type ‘Microsoft.Office.Interop.Wor…

ubuntu18.04 配置 mid360并测试fast_lio

1.在买到Mid360之后&#xff0c;我们可以看到mid360延伸出来了三组线。 第一组线是电源线&#xff0c;包含了红色线正极&#xff0c;和黑色线负极。一般可以用来接9-27v的电源&#xff0c;推荐接12v的电源转换器&#xff0c;或者接14.4v的电源转换器。 第二组线是信号线&#x…

CSS概述

CSS是一种样式表语言&#xff0c;用于为HTML文档控制外观&#xff0c;定义布局。例如&#xff0c; CSS涉及字体、颜色、边距、高度、宽度、背景图像、高级定位等方面 。 ● 可将页面的内容与表现形式分离&#xff0c;页面内容存放在HTML文档中&#xff0c;而用 于定义表现形式…

本学期嵌入式期末考试的综合项目,我是这么出题的

时间过得真快&#xff0c;临近期末&#xff0c;又到了老师出卷的时候。作为《嵌入式开发及应用》这门课的主讲教师&#xff0c;今年给学生出的题目有一点点难度&#xff0c;最后的综合项目要求如下所示&#xff0c;各位学生朋友和教师同行可以评论一下难度如何&#xff0c;单片…

【推荐算法】召回模型总结

文章目录 1、传统召回算法2、向量化召回统一建模架构2.1、如何定义正样本2.2、重点关注负样本2.3、召回生成Embedding&#xff1a;要求用户、物料解耦2.4、如何定义优化目标2.4.1、Softmax Loss、NCE Loss、NEG Loss2.4.2、Sampled Softmax Loss2.4.3、Pairwise Loss 3、Word2V…

量化交易入门——盘口

今天接着上一期讲解开盘定势的种类&#xff0c;在讲之前&#xff0c;科普一下“盘口五档”的成交知识。 每个炒股软件上&#xff0c;都会有某只个股的成交信息&#xff0c;在其中会出现一个五档的行情列表&#xff0c;里面列出了买家和卖家各五个价格及其对应的数量。这五档价…

Docker 基础使用(5)Compose

文章目录 Docker Compose 基础认识Docker Compose 基础语法Docker Compose 基础指令Docker Compose 使用实例 Docker 基础使用(0&#xff09;基础认识 Docker 基础使用(1&#xff09;使用流程概览 Docker 基础使用(2&#xff09;镜像与容器 Docker 基础使用(3&#xff09;存储卷…