当前位置: 首页 > news >正文

C语言——排序(冒泡,选择,插入)

基本概念

        排序是对数据进行处理的常见操作,即将数据按某字段规律排列。字段是数据节点的一个属性,比如学生信息中的学号、分数等,可针对这些字段进行排序。同时,排序算法有稳定性之分,若两个待排序字段一致的数据在排序前后相对位置不变,则该排序算法是稳定的,否则不稳定。此外,根据数据量与内存的关系,还分为内排序(数据量小,可全部装入内存处理)和外排序(数据量大,需暂存外存,分批次读入内存处理)。

一、冒泡排序

(一)核心思想

        冒泡排序基于相邻元素比较的思路。从头到尾让每两个相邻元素进行比较,若顺序符合排序要求则保持位置不变,若为逆序则交换位置。经过一轮比较,序列中具有 “极值”(最大值或最小值,取决于排序顺序)的数据会被挪至序列末端。

        例如,对于数组[5, 3, 8, 6, 2]进行升序排序,从第一个元素开始,5 和 3 比较,因为 5 > 3,所以交换位置,数组变为[3, 5, 8, 6, 2];接着 5 和 8 比较,顺序不变;8 和 6 比较,交换位置,数组变为[3, 5, 6, 8, 2];8 和 2 比较,交换位置,第一轮比较结束后数组变为[3, 5, 6, 2, 8]。可以看到,最大的数 8 被移到了数组末尾。若序列中有n个数据,在最极端情况下,经过n - 1轮比较一定能将所有数据排序完毕。

(二)代码实现

​
// 冒泡排序,data为数组,len为数组长度
void bubbleSort(int* data, int len) {int i, j;for (i = 0; i < len - 1; i++) {int flag = 1;  // 用于标记某一趟是否有元素交换,若没有则说明已排序完成for (j = 0; j < len - 1 - i; j++) {if (data[j] > data[j + 1]) {  // 升序排序,若要降序改为data[j] < data[j + 1]int tmp;tmp = data[j];data[j] = data[j + 1];data[j + 1] = tmp;flag = 0;  // 有元素交换,标记为0}}if (flag) {  // 若某一趟没有元素交换,提前结束排序break;}}
}​

(三)性能分析

        冒泡排序的时间复杂度为 O(n^2),其中n是待排序数据的数量。这是因为在最坏情况下,需要进行n - 1轮比较,每轮比较的次数从n - 1逐渐减少到 1。空间复杂度为 O(1),因为它只需要几个临时变量来进行元素交换,不需要额外的大量空间。冒泡排序是一种稳定的排序算法,因为相同元素的相对顺序在排序过程中不会改变。

(四)示例图示

冒泡排序

以数组[5, 3, 8, 6, 2]的升序排序为例:

第一轮:

  • 比较 5 和 3,交换,数组变为[3, 5, 8, 6, 2]
  • 比较 5 和 8,顺序不变
  • 比较 8 和 6,交换,数组变为[3, 5, 6, 8, 2]
  • 比较 8 和 2,交换,数组变为[3, 5, 6, 2, 8]

第二轮:

  • 比较 3 和 5,顺序不变
  • 比较 5 和 6,顺序不变
  • 比较 6 和 2,交换,数组变为[3, 5, 2, 6, 8]

第三轮:

  • 比较 3 和 5,顺序不变
  • 比较 5 和 2,交换,数组变为[3, 2, 5, 6, 8]

第四轮:

  • 比较 3 和 2,交换,数组变为[2, 3, 5, 6, 8],排序完成。

二、选择排序

(一)核心思想

        选择排序的思路是每一轮从待排序的数据元素中选出最小(或最大)的一个元素,存放在序列的起始位置,直到全部待排序的数据元素排完。

        例如,对于数组[5, 3, 8, 6, 2]进行升序排序,第一轮从数组中找到最小的数 2,与第一个数 5 交换位置,数组变为[2, 3, 8, 6, 5];第二轮从剩下的[3, 8, 6, 5]中找到最小的数 3,由于它已经在正确位置,无需交换;第三轮从[8, 6, 5]中找到最小的数 5,与 8 交换位置,数组变为[2, 3, 5, 6, 8];第四轮从[6, 8]中找到最小的数 6,由于它已经在正确位置,无需交换,此时数组已排序完成。

(二)代码实现

​
// 选择排序,data为数组,len为数组长度
void selectionSort(int* data, int len) {int i, j, minIndex;for (i = 0; i < len - 1; i++) {minIndex = i;for (j = i + 1; j < len; j++) {if (data[j] < data[minIndex]) {  // 升序排序,若要降序改为data[j] > data[minIndex]minIndex = j;}}if (minIndex != i) {int tmp = data[i];data[i] = data[minIndex];data[minIndex] = tmp;}}
}​

(三)性能分析

        选择排序的时间复杂度同样为 O(n^2),因为无论数据初始状态如何,都需要进行n - 1轮选择和交换操作。空间复杂度为 O(1),只需要几个临时变量。选择排序是一种不稳定的排序算法,例如数组[5, 5, 3]进行升序排序时,两个 5 的相对顺序可能会改变。

(四)示例图示

选择排序

以数组[5, 3, 8, 6, 2]的升序排序为例:

第一轮:

找到最小的 2,与 5 交换,数组变为[2, 3, 8, 6, 5]

第二轮:

在剩下的[3, 8, 6, 5]中找到最小的 3,位置不变

第三轮:

[8, 6, 5]中找到最小的 5,与 8 交换,数组变为[2, 3, 5, 6, 8]

第四轮:

[6, 8]中找到最小的 6,位置不变,排序完成。

三、插入排序

(一)核心思想

        插入排序假设前面已经有i个节点是有序的,那么从第i + 1个节点开始,插入到前面i个节点的合适位置中。由于第一个元素自身总是有序的,因此从第 2 个元素开始,不断插入前面的有序序列,直到全部排列完毕。

        例如,对于数组[5, 3, 8, 6, 2]进行升序排序,初始时认为第一个元素 5 是有序的。然后取第二个元素 3,将其与 5 比较,因为 3 < 5,所以将 5 后移一位,把 3 插入到 5 原来的位置,数组变为[3, 5, 8, 6, 2];接着取第三个元素 8,由于 8 > 5,所以 8 的位置不变,数组仍为[3, 5, 8, 6, 2];再取第四个元素 6,将其依次与 8、5 比较,找到合适位置插入,数组变为[3, 5, 6, 8, 2];最后取第五个元素 2,将其依次与 8、6、5、3 比较,找到合适位置插入,数组变为[2, 3, 5, 6, 8]

(二)代码实现

​
// 插入排序,data为数组,len为数组长度
void insertionSort(int* data, int len) {int i, j, tmp;for (i = 1; i < len; i++) {tmp = data[i];j = i - 1;while (j >= 0 && data[j] > tmp) {  // 升序排序,若要降序改为data[j] < tmpdata[j + 1] = data[j];j--;}data[j + 1] = tmp;}
}​

(三)性能分析

        插入排序在最好情况下(数据已经有序)的时间复杂度为 O(n),因为只需要进行n - 1次比较,无需移动元素。在最坏情况下(数据逆序)的时间复杂度为 O(n^2),平均时间复杂度也为O(n^2)。空间复杂度为 O(1),只需要几个临时变量。插入排序是一种稳定的排序算法,因为在插入过程中,相同元素的相对顺序不会改变。

(四)示例图示

插入排序

以数组[5, 3, 8, 6, 2]的升序排序为例:

第一轮:

3 插入到 5 前面,数组变为[3, 5, 8, 6, 2]

第二轮:

8 位置不变,数组仍为[3, 5, 8, 6, 2]

第三轮:

6 插入到 8 前面,数组变为[3, 5, 6, 8, 2]

第四轮:

2 插入到最前面,数组变为[2, 3, 5, 6, 8],排序完成。

总结

冒泡排序、选择排序和插入排序都是简单的排序算法,它们的时间复杂度在最坏情况下都O(n^2),空间复杂度为O(1),适用于数据样本较小的场合。其中,冒泡排序和插入排序是稳定的排序算法,选择排序是不稳定的排序算法。在实际应用中,可根据数据特点和需求选择合适的排序算法。

相关文章:

C语言——排序(冒泡,选择,插入)

基本概念 排序是对数据进行处理的常见操作&#xff0c;即将数据按某字段规律排列。字段是数据节点的一个属性&#xff0c;比如学生信息中的学号、分数等&#xff0c;可针对这些字段进行排序。同时&#xff0c;排序算法有稳定性之分&#xff0c;若两个待排序字段一致的数据在排序…...

git如何下载指定版本

要使用Git下载指定版本&#xff0c;可以通过以下步骤进行操作‌&#xff1a; ‌1. 使用Git命令行下载指定版本‌&#xff1a; 1.1 首先&#xff0c;使用git clone命令克隆整个git库到本地。例如&#xff1a;git clone [库的URL]。这将下载最新的代码到本地。‌ 1.2 进入克隆…...

数字电路-基础逻辑门实验

基础逻辑门是数字电路设计的核心元件&#xff0c;它们执行的是基本的逻辑运算。通过这些基本运算&#xff0c;可以构建出更为复杂的逻辑功能。常见的基础逻辑门包括与门&#xff08;AND&#xff09;、或门&#xff08;OR&#xff09;、非门&#xff08;NOT&#xff09;、异或门…...

新数据结构(9)——Java异常体系

异常的种类 程序本身通常无法主动捕获并处理错误&#xff08;Error&#xff09;&#xff0c;因为这些错误通常表示系统级的严重问题&#xff0c;但程序可以捕获并处理异常&#xff08;Excrption&#xff09;&#xff0c;而Error则被视为一种程序无法或不应尝试恢复的异常类型。…...

每日十题八股-补充材料-2025年2月15日

1.TCP是如何保证消息的顺序和可靠的&#xff1f; 写得超级好的文章 首先肯定是三次握手和四次挥手保证里通讯双方建立了正确有效的连接。 其次是校验和、序列号&#xff0c;ACK消息应答机制还有重传机制&#xff0c;保证了消息顺序和可靠。 同时配合拥塞机制和流量控制机制&am…...

使用 Python 爬虫获取微店快递费用 item_fee API 接口数据

在电商运营中&#xff0c;快递费用是影响商家利润和用户体验的重要因素之一。微店作为国内知名的电商平台&#xff0c;提供了丰富的 API 接口供开发者使用&#xff0c;其中也包括查询商品快递费用的接口。通过调用微店的 item_fee 接口&#xff0c;开发者可以获取指定商品的快递…...

通过用户名和密码登录服务器有哪些方法

通过用户名和密码登录到服务器的方式取决于你使用的工具和协议。以下是几种常见的方法&#xff1a; 1. 使用 SSH 登录到 Linux 服务器 你可以通过 SSH&#xff08;Secure Shell&#xff09;使用用户名和密码连接到远程服务器。通常&#xff0c;你会使用 ssh 命令来进行连接。…...

sort快排

当然可以!让我们通过类似的详细步骤来解释 快速排序(Quick Sort) 的原理和实现,就像之前解释 a &= (a - 1) 的原理一样。 快速排序(Quick Sort)原理 快速排序是一种高效的排序算法,其核心思想是分而治之。它通过选择一个“基准值”(pivot),将数组分为两部分: …...

用xml配置spring, bean标签有哪些属性?

用xml配置spring, bean标签有哪些属性? 在Spring框架中&#xff0c;使用XML配置文件时&#xff0c;<bean>标签用于定义一个Bean。以下是一些常用的<bean>标签属性&#xff1a; 1. class 描述&#xff1a;指定Bean的类名。示例&#xff1a;<bean id"myBe…...

纪念日倒数日项目的实现-【纪念时刻-时光集】

纪念日/倒数日项目的实现## 一个练手的小项目&#xff0c;uniappnodemysql七牛云。 在如今快节奏的生活里&#xff0c;大家都忙忙碌碌&#xff0c;那些具有特殊意义的日子一不小心就容易被遗忘。今天&#xff0c;想给各位分享一个“纪念日”项目。 【纪念时刻-时光集】 一…...

无人机不等同轴旋翼架构设计应用探究

“结果显示&#xff0c;对于不等组合&#xff0c;用户应将较小的螺旋桨置于上游以提高能效&#xff0c;但若追求最大推力&#xff0c;则两个相等的螺旋桨更为理想。” 在近期的研究《不等同轴旋翼性能特性探究》中&#xff0c;Max Miles和Stephen D. Prior博士深入探讨了不同螺…...

1-8 gitee码云的注册与使用

码云的网址&#xff1a;Gitee - 基于 Git 的代码托管和研发协作平台 这是一个国内的托管代码平台&#xff0c;速度要比国外的快 1.0 注册 如何注册码云&#xff1f; 查考文章&#xff1a;https://jingyan.baidu.com/article/425e69e6a8cad6ff14fc1615.html 2.0 使用 使用码云进…...

嵌入式硬件篇---OpenMV的硬件流和软件流

文章目录 前言一、硬件流控制&#xff08;Hardware Flow Control&#xff09;1. 基本原理RTSCTS 2. OpenMV中的实现• 硬件要求• 代码配置• 工作流程 二、软件流控制&#xff08;Software Flow Control&#xff09;1. 基本原理XONXOFF 2. OpenMV中的实现• 代码配置• 工作流…...

Word 里面嵌入DeepSeek

目录 一、问题描述 二、解决方法 三、代码 四、注意事项 五、总结 一、问题描述 如何在Word里面嵌入DeepSeek? 二、解决方法 1、新建文档&#xff0c;按 AltF11&#xff0c;进入VB界面。 2、选中文档&#xff0c;右键->插入->模块。 3、进入模块&#xff0c;粘入…...

聊聊 IP 地址和端口号的区别

在计算机网络中&#xff0c;两个基本概念对于理解设备如何通过网络进行通信至关重要。IP 地址和端口号是 TCP/IP 的典型特征&#xff0c;其定义如下&#xff1a;IP 地址是分配给连接到网络的每台机器的唯一地址&#xff0c;用于定位机器并与其通信。相反&#xff0c;端口号用于…...

rust学习一、入门之搭建简单开发环境

1、搭建开发环境(windows11&#xff09; a.登录官网 一看就明白&#xff0c;此处略。 b.安装rustup 一看就明白&#xff0c;此处略。 c.安装 cargo script 或者 rust-script script cargo install cargo-script 完成后 注意&#xff1a;时间有一点点久。 测试 cargo s…...

浅聊MQ之Kafka与RabbitMQ简用

&#xff08;前记&#xff1a;内容有点多&#xff0c;先看目录再挑着看。&#xff09; Kafka与RabbitMQ的使用举例 Kafka的使用举例 安装与启动&#xff1a; 从Apache Kafka官网下载Kafka中间件的运行脚本。解压后&#xff0c;通过命令行启动Zookeeper&#xff08;Kafka的运行…...

【原创】解决vue-element-plus-admin无法实现下拉框动态控制表单功能,动态显隐输入框

前言 目前使用vue-element-plus-admin想要做一个系统定时任务功能&#xff0c;可以选择不同的定时任务类型&#xff0c;比如使用cron表达式、周期执行、指定时间执行等。每种类型对应不同的输入框&#xff0c;需要动态显隐输入框才行&#xff0c;但是这个vue-element-plus-adm…...

SpringBoot开发——初步了解SpringBoot

文章目录 一、SpringBoot简介 1、什么是Spring Boot2、Spring Boot的优点3、Spring Boot功能 二、Spring与Spring Boot对比三、Spring Boot与Spring MVC四、Spring Boot体系结构五、Springboot Initializr 1、Spring Initializr2、Spring Initializr模块 一、SpringBoot简介…...

双轴伺服电机驱动控制器AGV、AMR专用双伺服电机驱动控制器解决方案

工业机器人数控机床XY机械手双轴机器人堆垛机专用双轴伺服电机驱动控制器48V 14ARMS带有STO功能&#xff0c;隔离高压CAN/RS485/USB通讯支持编码器和霍尔输入 双伺服电机驱动控制器TMCM2611功能介绍 集成2个伺服电机的控制和驱动于一体供电电压48V&#xff0c;驱动电流14A RM…...

【Web 进阶篇】优雅的接口设计:统一响应、全局异常处理与参数校验

系列回顾&#xff1a; 在上一篇中&#xff0c;我们成功地为应用集成了数据库&#xff0c;并使用 Spring Data JPA 实现了基本的 CRUD API。我们的应用现在能“记忆”数据了&#xff01;但是&#xff0c;如果你仔细审视那些 API&#xff0c;会发现它们还很“粗糙”&#xff1a;有…...

QT: `long long` 类型转换为 `QString` 2025.6.5

在 Qt 中&#xff0c;将 long long 类型转换为 QString 可以通过以下两种常用方法实现&#xff1a; 方法 1&#xff1a;使用 QString::number() 直接调用 QString 的静态方法 number()&#xff0c;将数值转换为字符串&#xff1a; long long value 1234567890123456789LL; …...

大学生职业发展与就业创业指导教学评价

这里是引用 作为软工2203/2204班的学生&#xff0c;我们非常感谢您在《大学生职业发展与就业创业指导》课程中的悉心教导。这门课程对我们即将面临实习和就业的工科学生来说至关重要&#xff0c;而您认真负责的教学态度&#xff0c;让课程的每一部分都充满了实用价值。 尤其让我…...

【数据分析】R版IntelliGenes用于生物标志物发现的可解释机器学习

禁止商业或二改转载&#xff0c;仅供自学使用&#xff0c;侵权必究&#xff0c;如需截取部分内容请后台联系作者! 文章目录 介绍流程步骤1. 输入数据2. 特征选择3. 模型训练4. I-Genes 评分计算5. 输出结果 IntelliGenesR 安装包1. 特征选择2. 模型训练和评估3. I-Genes 评分计…...

基于Java Swing的电子通讯录设计与实现:附系统托盘功能代码详解

JAVASQL电子通讯录带系统托盘 一、系统概述 本电子通讯录系统采用Java Swing开发桌面应用&#xff0c;结合SQLite数据库实现联系人管理功能&#xff0c;并集成系统托盘功能提升用户体验。系统支持联系人的增删改查、分组管理、搜索过滤等功能&#xff0c;同时可以最小化到系统…...

在QWebEngineView上实现鼠标、触摸等事件捕获的解决方案

这个问题我看其他博主也写了&#xff0c;要么要会员、要么写的乱七八糟。这里我整理一下&#xff0c;把问题说清楚并且给出代码&#xff0c;拿去用就行&#xff0c;照着葫芦画瓢。 问题 在继承QWebEngineView后&#xff0c;重写mousePressEvent或event函数无法捕获鼠标按下事…...

招商蛇口 | 执笔CID,启幕低密生活新境

作为中国城市生长的力量&#xff0c;招商蛇口以“美好生活承载者”为使命&#xff0c;深耕全球111座城市&#xff0c;以央企担当匠造时代理想人居。从深圳湾的开拓基因到西安高新CID的战略落子&#xff0c;招商蛇口始终与城市发展同频共振&#xff0c;以建筑诠释对土地与生活的…...

08. C#入门系列【类的基本概念】:开启编程世界的奇妙冒险

C#入门系列【类的基本概念】&#xff1a;开启编程世界的奇妙冒险 嘿&#xff0c;各位编程小白探险家&#xff01;欢迎来到 C# 的奇幻大陆&#xff01;今天咱们要深入探索这片大陆上至关重要的 “建筑”—— 类&#xff01;别害怕&#xff0c;跟着我&#xff0c;保准让你轻松搞…...

C++课设:简易日历程序(支持传统节假日 + 二十四节气 + 个人纪念日管理)

名人说:路漫漫其修远兮,吾将上下而求索。—— 屈原《离骚》 创作者:Code_流苏(CSDN)(一个喜欢古诗词和编程的Coder😊) 专栏介绍:《编程项目实战》 目录 一、为什么要开发一个日历程序?1. 深入理解时间算法2. 练习面向对象设计3. 学习数据结构应用二、核心算法深度解析…...

关于uniapp展示PDF的解决方案

在 UniApp 的 H5 环境中使用 pdf-vue3 组件可以实现完整的 PDF 预览功能。以下是详细实现步骤和注意事项&#xff1a; 一、安装依赖 安装 pdf-vue3 和 PDF.js 核心库&#xff1a; npm install pdf-vue3 pdfjs-dist二、基本使用示例 <template><view class"con…...