我知道你会冒泡排序,但是你会优化冒泡排序吗?

235 篇文章 1 订阅
235 篇文章 0 订阅

本文章出自【码同学软件测试
码同学公众号:自动化软件测试
码同学抖音号:小码哥聊软件测试

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

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

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

冒泡排序(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]

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

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

免费领取 码同学软件测试 课程笔记+超多学习资料+完整视频+最新面试题,可以转发文章 + 私信「码同学666」获取资料哦

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

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

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

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)。

最后祝大家学习愉快。

笔者跟大家分享一个福利!扫码回复【csdn码哥群】入软件测试自学交流群,可免费听技术讲座+领学习资料+视频课免费看
在这里插入图片描述
作者:码同学软件测试
本文著作权归作者所有,任何形式的转载都请联系作者获得授权并注明出处。

  • 0
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值