数据结构从入门到精通——冒泡排序

news2025/1/19 20:30:22

冒泡排序

  • 前言
  • 一、冒泡排序的基本思想
  • 二、冒泡排序的特性总结
  • 三、冒泡排序的动画演示
  • 四、冒泡排序的具体代码
    • test.c


前言

冒泡排序是一种简单的排序算法,通过重复遍历待排序数列,比较相邻元素的大小并交换位置,使得每一轮遍历后最大(或最小)的元素都会“冒泡”到数列的一端,直到整个数列有序。这种算法的时间复杂度较高,但在处理小规模数据或近乎有序的数据时表现良好,除此之外,与其他排序算法相比,冒泡排序更适用于教学而不适应于实际生活


一、冒泡排序的基本思想

在这里插入图片描述
冒泡排序的基本思想是通过相邻元素之间的比较和交换,使得每一趟排序过程中,最大(或最小)的元素能够“冒泡”到序列的一端,从而达到排序的目的。

具体来说,冒泡排序的算法过程可以分为以下几个步骤:

  1. 从序列的第一个元素开始,比较相邻的两个元素,如果它们的顺序错误(即前一个元素大于后一个元素),则交换这两个元素的位置。
  2. 接着,从序列的第二个元素开始,重复上述步骤,直到序列的最后一个元素。这样,第一趟排序结束后,最大的元素就会被放到序列的最后面。
  3. 接下来,对序列的前n-1个元素进行同样的操作,直到整个序列都有序为止。在每一趟排序过程中,都会有一个最大(或最小)的元素被放到序列的末尾。

冒泡排序的时间复杂度为O(n^2),其中n为序列的长度。虽然它的效率不如一些更高级的排序算法,但由于其实现简单,易于理解,因此在一些实际应用中仍然被广泛使用。

例如,在一些小型数据集的排序中,冒泡排序可以作为一种简单有效的解决方案。此外,在一些需要稳定排序的场合中,冒泡排序也是一种不错的选择,因为它在交换相邻元素时不会改变相等元素的相对顺序。但是像上述这些情况生活中出现的概率很小,没有人会为了冒泡排序而专门设置一串代码。

总之,冒泡排序虽然不是最优的排序算法,但它的基本思想简单易懂,具有教学意义,不适用于实际生活。

二、冒泡排序的特性总结

  1. 冒泡排序是一种非常容易理解的排序
  2. 时间复杂度:O(N^2)
  3. 空间复杂度:O(1)
  4. 稳定性:稳定

冒泡排序,作为一种基础的排序算法,虽然在实际应用中由于其效率问题较少被直接使用,但在理解排序算法的基本原理和特性上,它起到了至关重要的作用。以下是对冒泡排序特性的总结:

  1. 稳定性:冒泡排序是一种稳定的排序算法。这意味着在排序过程中,对于相等的元素,它们的相对顺序不会发生改变。这对于某些需要保持原有数据结构中元素间关系的场景来说是非常重要的。

  2. 简单易懂:冒泡排序的实现逻辑相对直观,容易理解。它通过相邻元素之间的比较和交换来逐步将最大值或最小值“冒泡”到序列的一端。

  3. 效率问题:尽管冒泡排序在理解上较为简单,但其效率并不高。它的时间复杂度在最坏情况下为O(n^2),其中n为待排序序列的长度。这意味着对于大型数据集,冒泡排序可能不是最优的选择。

  4. 优化空间:尽管基本冒泡排序效率不高,但可以通过一些优化手段来改进其性能。例如,可以设置一个标志位来跟踪在一次完整的遍历过程中是否发生了交换,如果没有发生交换,则说明序列已经有序,可以提前结束排序。

  5. 适应性:冒泡排序适用于小规模数据集或者部分有序的序列。对于部分有序的序列,冒泡排序的效率会相对较高,因为其可以在较少的遍历次数内完成排序。

综上所述,冒泡排序虽然在实际应用中可能不是最优的选择,但它在教学、理解排序算法原理以及处理小规模数据集或特定场景(如稳定性要求高的场景)下仍然具有重要意义。通过深入理解冒泡排序的特性,我们可以更好地掌握排序算法的基本原理和优化方法,为处理更复杂的数据结构和算法问题打下坚实的基础。

三、冒泡排序的动画演示

冒泡排序

冒泡排序的动画演示展示了冒泡排序算法的工作过程。在演示中,可以看到一系列数字按照顺序逐个比较和交换位置,直到所有数字按照升序或降序排列。通过动画,可以清晰地看到每个步骤中数字的变化,从而理解冒泡排序算法的原理和步骤。这种演示方式有助于学习者更好地掌握冒泡排序算法,并理解其在实际应用中的工作原理。

四、冒泡排序的具体代码

test.c

#include<stdio.h>
#include<stdlib.h>
#include<time.h>

void BubbleSort(int* a, int n);
void PrintArray(int* a, int n);
void Swap(int* p1, int* p2)
{
	int tmp = *p1;
	*p1 = *p2;
	*p2 = tmp;
}
void TestBubbleSort()
{
	//int a[] = { 5, 13, 9, 16, 12, 4, 7, 1, 28, 25, 3, 9, 6, 2, 4, 7, 1, 8 };
	int a[] = { 5, 3, 9, 6, 2, 4, 7, 1, 8 };
	PrintArray(a, sizeof(a) / sizeof(int));

	BubbleSort(a, sizeof(a) / sizeof(int));

	PrintArray(a, sizeof(a) / sizeof(int));
}


void PrintArray(int* a, int n)
{
	for (int i = 0; i < n; i++)
	{
		printf("%d ", a[i]);
	}
	printf("\n");
}
void BubbleSort(int* a, int n)
{
	for (int i = 0; i < n - 1; i++)
	{
		int exchange = 0;
		for (int j = 0; j < n - i - 1; j++)
		{
			if (a[j + 1] <= a[j])
			{
				Swap(&a[j + 1], &a[j]);
				exchange = 1;
			}
		}
		if (exchange == 0)break;
	}
}
void TestOP()
{
	srand(time(0));
	const int N = 10000;
	int* a1 = (int*)malloc(sizeof(int) * N);

	for (int i = 0; i < N; ++i)
	{
		a1[i] = rand();
	}
	int begin1 = clock();
	BubbleSort(a1, N);
	int end1 = clock();
	printf("BubbleSort:%d\n", end1 - begin1);

	free(a1);
}

int main()
{
	TestBubbleSort();
	TestOP();

	return 0;
}

这段代码实现了冒泡排序算法。冒泡排序的基本思想是通过相邻元素的比较和交换来将大的元素逐步“冒泡”到最后。

代码中的函数BubbleSort接受两个参数,一个是待排序数组a,另一个是数组的长度n。

首先,外层的循环i表示排序的轮数,每一轮会把当前未排序部分的最大元素冒泡到最后。循环的终止条件是i < n - 1,因为最后一个元素无需再进行比较。

内层的循环j用来进行相邻元素的比较和交换。每一轮的比较从数组的第一个元素开始,依次比较相邻的两个元素。如果后一个元素小于等于前一个元素,就交换它们的位置。

在每一轮的内层循环结束后,通过exchange变量来判断是否有元素发生了交换。如果没有发生交换,说明数组已经是有序的,就可以提前结束排序。

最终,当外层的循环结束后,整个数组就按照从小到大的顺序排列好了。
在这里插入图片描述


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

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

相关文章

ubuntu内存不足,用Swap扩展增加虚拟内存

Linux增大Swap分区&#xff0c;可以增加虚拟内存&#xff0c;以解决电脑卡机&#xff0c;内存不足等问题 top可以查看cpu的使用情况 lscpu可以查看本机配置的cpu硬件情况 查看内存使用情况 free -h (下面显示"交换"或者Swap等字样说明系统已经启动了Swap&#xff…

EMC Unity存储系统(包含VNXe)常用检查命令

DELL EMC的Unity存储系统&#xff0c;包括VNXe存储系统的OS已经完全和Clariion 的VNX不同了&#xff0c;近期遇到很多关于EMC unity存储系统故障的一些初步检查需求&#xff0c;下面是一些对于DELL EMC Unity存储系统的最常用的底层检查命令&#xff0c;可以对系统故障有个初步…

深度学习——线性代数相关知识

线性代数基础知识 一、线性代数基础知识1、标量2、向量3、矩阵4、张量5、点积6、向量—矩阵积7、矩阵—矩阵乘法 二、小结 一、线性代数基础知识 本节将介绍简要地回顾一下部分基本线性代数内容&#xff0c;线性代数中的基本数学对象、算术和运算&#xff0c;并用数学符号和相…

Linux编程4.11 网络编程-广播

广播实现一对多的通讯 它通过向广播地址发送数据报文实现的 1、套接字选项 套接字选项用于修饰套接字以及其底层通讯协议的各种行为。函数setsockopt和getsockopt可以查看和设置套接字的各种选项。 #include <sys/types.h> #include <sys/socket.h>int getso…

申请免费IP地址证书

目录 IP申请SSL证书需要满足什么条件呢&#xff1f; 为什么需要申请IP地址证书&#xff1f; 支持IP地址SSL证书类型 DV级别IP SSL证书和OV级别IP SSL证书的区别 申请公网IP地址证书有免费的吗&#xff1f; 背景&#xff1a;当用户直接通过IP地址而非域名访问网站时&#xf…

后端如何返回404地址

当我们网站输入不存在的地址&#xff0c;经常会出现404的页面&#xff0c;这是如何做到的 1.添加配置 spring:mvc:view:prefix: /templates/suffix: .html 2.resources下添加templates目录&#xff0c;下面放404的网站 3.添加依赖&#xff0c;版本在主pom里面配置好了&#x…

霍格沃兹测试开发从人员外包到测试工具、测试平台,提供全方位的测试解决方案~

随着学社的学员越来越多&#xff0c;影响力越来越大&#xff0c;不停有学员和企业问我们&#xff1a;能否提供人员外包服务&#xff1f;与此同时&#xff0c;企业对于外包人员的业务技能要求也越来越高&#xff0c;寻找一个稳定靠谱的供应商也成了很多学员所在公司的需求。对此…

web前端常用标签(html)

1.定义 1.1标签 语法规范&#xff1a;<标签名 属性名"属性值">标签名</标签名> 标签之间可以嵌套 1.2属性 定制元素的行为的。属性是不通用的&#xff0c;每一个标签存在自身的属性。当属性名属性值时&#xff0c;可以只写属性值 2.HTML常用标签 2…

【人工智能Ⅱ】实验2:VGG图像分类

实验2&#xff1a;VGG图像分类 一&#xff1a;实验目的与要求 1&#xff1a;掌握VGG网络的原理与结构。 2&#xff1a;学会利用VGG网络建立训练模型&#xff0c;并对模型进行评估。 3&#xff1a;学会使用VGG网络进行分类。 二&#xff1a;实验内容 1&#xff1a;用VGG网络…

扩展欧拉定理

为了求a^bmodm的余数,我们可以利用扩展欧拉定理给出的同余方程,转化成一个好求的式子,首先我们要能够解出欧拉函数.欧拉函数讲解可以看看这篇欧拉函数最全总结-CSDN博客(原理我不懂,只会用) 下面给出代码 using i64 long long; int phi(int n) {int res n;for (int i 2; i…

【C语言】内存管理内存管理函数文件管理文件管理函数

主页&#xff1a;醋溜马桶圈-CSDN博客 专栏&#xff1a;C语言_醋溜马桶圈的博客-CSDN博客 gitee&#xff1a;mnxcc (mnxcc) - Gitee.com 目录 1.C/C程序的内存开辟 2.内存相关的函数 2.1 memcpy 2.1.1 memcpy函数的使用 2.1.2 memcpy函数的模拟实现 2.2 memmove 2.2.2 me…

四阶Runge-Kutta方法求解高阶微分方程

一、经典的Runge-Kutta方法&#xff08;四级四阶RK方法&#xff09; Runge-Kutta法&#xff08;简写为RK方法&#xff09;既可达到较高精度&#xff0c;又可避免高阶导数计算。 对微分方程,在区间上的四阶Runge-Kutta方法的公式如下&#xff1a; 二、利用4阶Runge-Kutta方法计…

Elasticsearch - Docker安装Elasticsearch8.12.2

前言 最近在学习 ES&#xff0c;所以需要在服务器上装一个单节点的 ES 服务器环境&#xff1a;centos 7.9 安装 下载镜像 目前最新版本是 8.12.2 docker pull docker.elastic.co/elasticsearch/elasticsearch:8.12.2创建配置 新增配置文件 elasticsearch.yml http.host…

.locked勒索病毒是什么,企业数据被加密了如何恢复?

.locked勒索病毒介绍 .locked勒索病毒是一种恶意软件&#xff0c;它利用加密技术锁定用户的数据或系统&#xff0c;并以此进行勒索。用户一旦感染此病毒&#xff0c;将无法访问其重要文件&#xff0c;病毒会要求用户支付一笔赎金以获取解密密钥。这种病毒通常使用强大的加密算法…

为什么选VR全景技术进行乡村展示,VR全景技术助力乡村振兴

引言&#xff1a; 在科技飞速发展的当下&#xff0c;乡村振兴成为国家重要战略&#xff0c;如何创新性地展示乡村特色&#xff0c;提升乡村吸引力&#xff0c;成为当务之急。VR全景技术&#xff0c;作为一种新兴的展示手段&#xff0c;可以为乡村展示提供全新的视角&#xff0…

可观测性体系建设后,该如何挖掘数据及工具价值?

在现代企业的运维管理中&#xff0c;构建高效且可靠的可观测性体系是保障系统稳定性和业务连续性的关键。然而&#xff0c;运维团队成员的技术能力参差不齐往往成为实现这一目标的障碍。尤其在处理复杂系统故障时&#xff0c;高度依赖专业知识和经验的可观测性工具很难被全员有…

如何在Linux Ubuntu系统安装Nginx服务并实现无公网IP远程连接

文章目录 1. 安装Docker2. 使用Docker拉取Nginx镜像3. 创建并启动Nginx容器4. 本地连接测试5. 公网远程访问本地Nginx5.1 内网穿透工具安装5.2 创建远程连接公网地址5.3 使用固定公网地址远程访问 在开发人员的工作中&#xff0c;公网远程访问内网是其必备的技术需求之一。对于…

vue key的bug

今天遇到一个bug&#xff0c;列表删除元素时&#xff0c;明明在外层设置了key&#xff0c;但是列表元素的状态居然复用了&#xff0c;找了好久原因&#xff0c;最后是key的取值问题&#xff0c;记录一下。 首先key可以取undefine&#xff0c;这个是不会报错的 然后项目的代码结…

C#配置连接数据库字段

在Web.config文件中 添加如下配置 <!--连接数据库字段--><connectionStrings><add name"sql" connectionString"server.;uidsa;pwd8888;databaseArticleWebSite" /></connectionStrings>

element plus等框架中属性值是组件如何传入,替换分页图标

在 Vue 中替换element plus 分页图标 正常写法引入组件 import prevIcon from /components/xx.vue;<el-pagination layout"prev, pager, next" :prev-icon"prevIcon" :total"5" />利用 h 函数写法 const prevIcon h(div, [xr]);可以写…