七大经典排序算法——冒泡排序

文章目录

  • 📑冒泡排序介绍
  • 🌤️代码实现
  • 🌤️做个简单的优化
  • 🌤️复杂度和稳定性分析
  • ☁️结语


📑冒泡排序介绍

冒泡排序是一种简单但效率较低的排序算法。它重复地比较相邻的两个元素,如果顺序不对则交换它们,直到所有元素都被比较过一次。这样每一轮比较过后,最大的元素就会"冒泡"到最后面。接着,算法将会忽略最后一个元素,重复上述比较和交换的过程,直到所有元素都按照顺序排列。

⭐ 冒泡排序算法的步骤如下:

  1. 从数组的第一个元素开始,依次比较相邻的两个元素。
  2. 如果前一个元素大于后一个元素,交换它们的位置。
  3. 继续比较下一对相邻元素,直到最后一个元素。
  4. 重复以上步骤,每一轮比较都会将最大的元素"冒泡"到最后面。
  5. 当比较结束时,此时数组已经排好序,排序结束。

动图演示如下:
在这里插入图片描述

🌤️代码实现

🌳C语言版

// 交换函数,用于交换数组中两个元素的位置
void swap(int* arr, int i, int j) {int tmp = arr[i];  // 将第一个元素的值存储在临时变量 tmp 中arr[i] = arr[j];   // 将第二个元素的值赋给第一个元素的位置arr[j] = tmp;      // 将临时变量 tmp 的值赋给第二个元素的位置
}// 冒泡排序函数,用于对数组进行排序
void bobbleSort(int* arr, int n) {// 外层循环控制需要遍历的轮数,共进行 n-1 次遍历for (int i = 0; i < n - 1; i++) {// 内层循环进行相邻元素的比较和交换// j 的范围是从 0 到 n-i-1,这样每次内层循环可以比较并将最大值移动到最后for (int j = 0; j < n - i - 1; j++) { // j 进循环的条件容易出错,特殊标记下// 如果前一个元素比后一个元素大,交换它们if (arr[j] > arr[j + 1]) {swap(arr, j, j + 1);  // 调用交换函数进行交换}}}
}

🌳java版

// 交换函数,用于交换数组中两个元素的位置
private static void swap(int[] array, int i, int j) {int tmp = array[i]; // 将第一个元素的值存储在临时变量 tmp 中array[i] = array[j]; // 将第二个元素的值赋给第一个元素的位置array[j] = tmp; // 将临时变量 tmp 的值赋给第二个元素的位置
}// 冒泡排序函数,用于对数组进行排序
public static void bubbleSort(int[] array) {// 外层循环控制需要遍历的轮数for (int i = 0; i < array.length; i++) {// 内层循环进行相邻元素的比较和交换for (int j = i + 1; j < array.length; j++) {// 如果前一个元素比后一个元素大,交换它们if (array[i] > array[j]) {swap(array, i, j); // 调用交换函数进行交换}}}
}

🌤️做个简单的优化

可以在内层循环中定义一个变量 flag:用于检测这一轮内层循环是否发生了交换。初始值设为 1。
如果在一轮内层循环中没有发生交换(flag 仍为 1),说明数组已经有序,跳出循环即可。
🌳C语言版

// 交换函数,用于交换数组中两个元素的位置
void swap(int* arr, int i, int j) {int tmp = arr[i];  // 将第一个元素的值存储在临时变量 tmp 中arr[i] = arr[j];   // 将第二个元素的值赋给第一个元素的位置arr[j] = tmp;      // 将临时变量 tmp 的值赋给第二个元素的位置
}// 冒泡排序函数,用于对数组进行排序
void bobbleSort(int* arr, int n) {// 外层循环控制需要遍历的轮数,共进行 n-1 次遍历for (int i = 0; i < n - 1; i++) {int flag = 1; // 标记用于检测这一轮是否发生了交换// 内层循环进行相邻元素的比较和交换// j 的范围是从 0 到 n-i-1,这样每次内层循环可以比较并将最大值移动到最后for (int j = 0; j < n - i - 1; j++) { // j 进循环的条件容易出错,特殊标记下// 如果前一个元素比后一个元素大,交换它们if (arr[j] > arr[j + 1]) {swap(arr, j, j + 1); // 调用交换函数进行交换flag = 0; // 发生交换后,将标记设置为 0}}// 如果没有发生交换,说明数组已经有序,提前退出排序if (flag) {break;}}
}

🌳java版


// 交换函数,用于交换数组中两个元素的位置
private static void swap(int[] array, int i, int j) {int tmp = array[i];  // 将第一个元素的值存储在临时变量 tmp 中array[i] = array[j]; // 将第二个元素的值赋给第一个元素的位置array[j] = tmp;      // 将临时变量 tmp 的值赋给第二个元素的位置
}public static void bubbleSort(int[] array) {// 外层循环控制需要遍历的轮数,共进行 array.length 次遍历for (int i = 0; i < array.length; i++) {boolean flg = true; // 标记用于检测这一轮是否发生了交换// 内层循环进行相邻元素的比较和交换// j 的范围是从 i+1 到 array.length,注意这里的起始点是 i+1for (int j = i + 1; j < array.length; j++) {// 如果前一个元素比后一个元素大,交换它们if (array[i] > array[j]) {swap(array, i, j); // 调用交换函数进行交换flg = false; // 发生交换后,将标记设置为 false}}// 如果没有发生交换,说明数组已经有序,提前退出排序if (flg) {break;}}
}

🌤️复杂度和稳定性分析

时间复杂度分析:在最坏的情况下,冒泡排序需要进行n-1轮比较,每轮比较需要进行n-i次。因此,总的比较次数为(n-1) + (n-2) + … + 2 + 1 = n(n-1)/2,近似为O(n2)。

空间复杂度分析:使用了常数个变量,因此空间复杂度为O(1)

什么是稳定性?
答:稳定性指的是相同的数据所在的位置经过排序后是否发生变化。换句话说就是大小相同的两个值在排序之前和排序之后的先后顺序不变,这就是稳定的。

稳定性分析:冒泡排序将小的元素往前调或者把大的元素往后调;比较的是相邻的两个元素,交换也发生在这两个元素之间;因为相等的元素不会进行交换,所以稳定。

总结: 冒泡排序的时间复杂度为O(N2),空间复杂度为O(1),而且是稳定的排序。


☁️结语

请给自己些耐心,不要急于求成。
山外青山楼外楼,莫把百尺当尽头。
保持空杯心态加油努力吧!


都看到这里啦!真棒(*^▽^*)

可以给作者一个免费的赞赞吗,这将会鼓励我继续创作,谢谢大家

如有纰漏或错误,欢迎指正


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

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

相关文章

C++ socket epoll IO多路复用

IO多路复用通常用于处理单进程高并发&#xff0c;在Linux中&#xff0c;一切皆文件&#xff0c;一个socket连接会对应一个文件描述符&#xff0c;在监听多个文件描述符的状态应用中epoll相对于select和poll效率更高 epoll本质是系统在内核维护了一颗红黑树&#xff0c;监听的文…

Linux中bash脚本怎么表示一个字符串变量

Linux中bash脚本怎么表示一个字符串变量 在Bash脚本中&#xff0c;你可以使用单引号&#xff08;&#xff09;或双引号&#xff08;"&#xff09;来表示一个字符串变量。以下是两种方式的示例&#xff1a; 使用单引号&#xff08;&#xff09;&#xff1a; my_variable…

flink 和 clipper搭配使用

Flink是一个用于流处理和批处理的开源框架&#xff0c;可以实时数据处理和分析。 Clipper 是一个用于机器学习模型服务化的开源框架&#xff0c;能够轻松部署和管理机器学习模型&#xff0c;使模型可以通过统一的接口提供在线推理服务。 flink和clipper搭配使用&#xff1a; …

Leetcode | 5-21| 每日一题

2769. 找出最大的可达成数字 考点: 暴力 数学式子计算 思维 题解 通过式子推导: 第一想法是二分确定区间在区间内进行查找是否符合条件的, 本题最关键的便是 条件确定 , 第二种方法: 一般是通过数学公式推导的,这种题目我称为数学式编程题 代码 条件判断式 class Solution { …

需求分析的任务

1 确定对系统的综合要求 虽然功能需求是对软件系统的一项基本需求&#xff0c;但却并不是唯一的需求。通常对软件系统有下述几方面的综合要求。 1&#xff0e;功能需求 这方面的需求指定系统必须提供的服务。通过需求分析应该划分出系统必须完成的所有功能。 2&#xff0e;性能…

MacBook 怎么玩Windows游戏 苹果笔记本怎么玩游戏?mac上如何玩windows游戏

传统上&#xff0c;Mac 不被认为是好的游戏机。然而&#xff0c;苹果已经开始在 Mac 上的游戏上投入更多精力&#xff0c;特别是自从转向苹果芯片以来。这使得 Mac 游戏的本机移植数量和模拟 Windows 游戏的能力都得到了显著提高。 方法一&#xff1a;Boot Camp 1、Boot Camp是…

SpirngMVC框架学习笔记(一):SpringMVC基本介绍

1 SpringMVC 特点&概述 SpringMVC 从易用性&#xff0c;效率上 比曾经流行的 Struts2 更好 SpringMVC 是 WEB 层框架&#xff0c;接管了 Web 层组件, 比如控制器, 视图, 视图解析, 返回给用户的数据格式, 同时支持 MVC 的开发模式/开发架构SpringMVC 通过注解&#xff0c;…

Java数据结构和算法(B树)

前言 B树又叫平衡的多路搜索树&#xff1b;平衡的意思是又满足平衡二叉树的一些性质&#xff0c;左树大于右树&#xff1b; 多路意思是&#xff0c;可以多个结点&#xff0c;不再是像二叉树只有两个结点&#xff1b; 实现原理 B树是一种自平衡的搜索树&#xff0c;通常用于实…

MySQL和MongoDB数据库的区别

MySQL和MongoDB数据库的区别 随着大数据和云计算技术的兴起&#xff0c;数据库的选择成为开发者和架构师必须面对的重要决策。MySQL和MongoDB作为关系型数据库和非关系型数据库的代表&#xff0c;在各自领域都有着广泛的应用。本文将从多方面详细比较MySQL和MongoDB&#xff0…

MATLAB:插值函数之interp与griddata

MATLAB 提供了多种插值函数来处理不同维度的数据。其中&#xff0c;interp1、interp2 和 griddata 是常用的插值函数&#xff0c;分别用于一维、二维和多维&#xff08;不规则&#xff09;数据的插值。 之前有对interp1进行过详细介绍&#xff0c;如需详细了解&#xff0c;请查…

会声会影调速怎么用 会声会影如何调整音频速度

会声会影是一款功能强大的视频编辑软件&#xff0c;可以帮助我们轻松的实现剪辑。 会声会影的操作简单易懂&#xff0c;界面简洁明快。适合家庭使用&#xff0c; 我们使用会声会影可以在家就能将视频剪辑成好莱坞大片。但是在使用的过程中&#xff0c;仍然会遇到一些操作上的问…

洛谷 P3803 【模板】多项式乘法(FFT)

【模板】多项式乘法&#xff08;FFT&#xff09; 题目背景 这是一道多项式乘法模板题。 注意&#xff1a;本题并不属于中国计算机学会划定的提高组知识点考察范围。 题目描述 给定一个 n n n 次多项式 F ( x ) F(x) F(x)&#xff0c;和一个 m m m 次多项式 G ( x ) G(…

C语言--指针数组和数组指针的区别

指针数组 就是一个数组&#xff0c;由指针构成的数组&#xff0c;每一个元素都是指针&#xff0c;每个指针可以指向不同的内存地址&#xff0c;这些地址可以是数组、变量。 int var1 10; int var2 20; int var3 30;int *ptrArray[3]; // 定义一个指针数组&#xff0c;包含…

2024年上半年软件系统架构师论文【回忆版】

文章目录 考试时间考试地点案例分析1、微服务架构的优点和缺点2、质量属性的6个元素3、分布式锁 Redis的缺点4、MongoDB 存储矢量图的优势 论文回忆版论文一、论单元测试的设计与应用论文二、论大数据模型的设计与应用论文三、论模型驱动的架构设计及应用论文四、论云原生运维的…

Mybatis-Plus-Join

1. 简介 官网 https://mybatisplusjoin.com/ 2. 基本用法 步骤&#xff1a; 添加依赖 <!--mybatis-plus-join--> <dependency><groupId>com.github.yulichang</groupId><artifactId>mybatis-plus-join-boot-starter</artifactId><ve…

探索LangGraph:如何创建一个既智能又可控的航空客服AI

这种设计既保持了用户控制权&#xff0c;又确保了对话流程的顺畅。但随着工具数量的增加&#xff0c;单一的图结构可能会变得过于复杂。我们将在下一节中解决这个问题。 第三部分的图将类似于下面的示意图&#xff1a; 状态定义 首先&#xff0c;定义图的状态。我们的状态和L…

homography原理和图像相似度计算

1. homography 讲homography原理 讲homography应用 2. 图像相似度计算 20230621-计算两幅图像的相似度 20221205-有史以来最全的图像相似度算法 20231112-图像相似度对比方法

C++:List的使用和模拟实现

✨✨✨学习的道路很枯燥&#xff0c;希望我们能并肩走下来! 文章目录 目录 文章目录 前言 一 list的介绍及使用 1.1 list的介绍 1.2 list的使用 1.2.1 list的构造 1.2.2 list iterator的使用 1.2.3 list capacity 1.2.4 list element access 1.2.5 list modifiers …

golang+redis的延时队列

网址 https://github.com/cfanbo/delay-queue-redis 代码结构很简单&#xff0c;简单代表着自由度很高&#xff0c;使用过程中出现问题也很好修改。 我很喜欢这样的代码&#xff0c;至少我看的懂&#xff0c;该有的都有。 //package main // //import ( // "context&q…