IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> 我知道你会冒泡排序,但是你会优化冒泡排序吗? -> 正文阅读

[数据结构与算法]我知道你会冒泡排序,但是你会优化冒泡排序吗?

在生活中,我们离不开排序,比如我们上学的时候按个头高低排位置,现在我们买东西的时候会按照发货地远近进行排序,或者价格高低排序。

排序看着简单,可是背后藏着很多的算法和思想。在这给大家介绍一下常用的排序算法。

每次提到排序,绕不开的就是冒泡排序。

冒泡排序(Bubble sort)是一种基础的交换排序。它的基本思想是:两两比较相邻记录的关键字,如果反序则交换,直到没有反序的记录为止。这个算法的名字由来是因为越小的元素会经由交换慢慢“浮”到数列的顶端(升序或降序排列),就如同碳酸饮料中二氧化碳的气泡最终会上浮到顶端一样,故名“冒泡排序”。

我们先来看一个例子。
在这里插入图片描述

有这么8个数字组成的无序数列[5,8,6,3,9,2,1,7],希望按照从小到大的顺序对其进行排序。

按照冒泡排序的算法,我们需要链路比较相邻纪录的关键字,如果一个元素大于右边相邻元素时,交换他们的位置,当一个元素小于或等于右边元素,位置不变。
在这里插入图片描述

第一趟下来,元素9作为这个数列中最大的元素,就像是汽水里的气泡一样,冒到了最右侧。

再来第二趟:
在这里插入图片描述

第二次交换结束,8作为未排序中的最大元素被冒到右侧第二位置上。

将未排序的都遍历过之后,所有的元素都会变为有序,这就是冒泡排序的思路。

接下来,我们来实现这个代码,代码非常简单,使用双循环实现。内部循环控制每一轮的最大元素冒泡处理,外部循环控制所有元素的依次排序。

def bubble_sort(nums): for i in range(len(nums) - 1): # 这个循环负责设置冒泡排序进行的次数 for j in range(len(nums) - i - 1): # 内部循环控制每一轮的最大元素冒泡处理 if nums[j] > nums[j + 1]: nums[j], nums[j + 1] = nums[j + 1], nums[j] return numsprint(bubble_sort([5,8,6,3,9,2,1,7]))# 输出:[1, 2, 3, 5, 6, 7, 8, 9]

我们来看看冒泡排序的基本性能:
在这里插入图片描述

关于稳定性:因为在比较的过程中,当两个相同大小的元素相邻,只比较大或者小,所以相等的时候是不会交换位置的。而当两个相等元素离着比较远的时候,也只是会把他们交换到相邻的位置。他们的位置前后关系不会发生任何变化,所以算法是稳定的。

冒泡排序是一种比较好理解的排序,但是整体的性能相比较来说比较差,因此需要进行优化。

我们来看一个特殊情况。
在这里插入图片描述

如果按照上面说的算法去进行计算,我们加一点代码进去看看具体排序情况:

def bubble_sort(nums):

for i in range(len(nums) - 1): # 这个循环负责设置冒泡排序进行的次数

for j in range(len(nums) - i - 1): # 内部循环控制每一轮的最大元素冒泡处理

if nums[j] > nums[j + 1]:

nums[j], nums[j + 1] = nums[j + 1], nums[j]

print(“第”, i , “次排序后:”, end=’’)

for n in nums:

print(n, end=’ ,’)

print(" ")

return nums

print(bubble_sort([1,3,4,5,6,7,8,9]))

实际输出:

第 0 次排序后:1 ,3 ,4 ,5 ,6 ,7 ,8 ,9 ,

第 1 次排序后:1 ,3 ,4 ,5 ,6 ,7 ,8 ,9 ,

第 2 次排序后:1 ,3 ,4 ,5 ,6 ,7 ,8 ,9 ,

第 3 次排序后:1 ,3 ,4 ,5 ,6 ,7 ,8 ,9 ,

第 4 次排序后:1 ,3 ,4 ,5 ,6 ,7 ,8 ,9 ,

第 5 次排序后:1 ,3 ,4 ,5 ,6 ,7 ,8 ,9 ,

第 6 次排序后:1 ,3 ,4 ,5 ,6 ,7 ,8 ,9 ,

[1, 3, 4, 5, 6, 7, 8, 9]

我们能看到在第一次循环的时候,整个数列已经是有序的,但是常规版的算法依然会继续流程,浪费很多的时间,而这些都是多余的。

如果我们能判断出数列已经有序,并且做出标记,那么剩下的几轮排序就不必执行了,可以提前结束工作。我们来改良一下代码:

def bubble_sort(nums):

for i in range(len(nums) - 1): # 这个循环负责设置冒泡排序进行的次数

#有序标记,每一轮的初始值都是True

is_sorted = True

for j in range(len(nums) - i - 1): # 内部循环控制每一轮的最大元素冒泡处理

if nums[j] > nums[j + 1]:

#因为有元素进行交换,所以不是有序的,标记变为False

nums[j], nums[j + 1] = nums[j + 1], nums[j]

is_sorted = False

print(“第”, i , “次排序后:”, end=’’)

for n in nums:

print(n, end=’ ,’)

print(" ")

if is_sorted:

break

return nums

print(bubble_sort([1,3,4,5,6,7,8,9]))

输出:[1, 3, 4, 5, 6, 7, 8, 9]

加上这个标记位之后,我们再来运行一下:

第 0 次排序后:1 ,3 ,4 ,5 ,6 ,7 ,8 ,9 ,

[1, 3, 4, 5, 6, 7, 8, 9]

可以看到,当整体数列已经有序的基础上,不需要再进行后续的多次排序时间浪费了。整体的最好时间就优化为O(n)。

最后祝大家学习愉快。

  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2021-11-18 11:25:30  更:2021-11-18 11:26:11 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/9 2:11:16-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码