跟着动画学 Go 数据结构之冒泡排序

简介: 冒泡排序是一种最简单的交换排序算法。什么是交换?交换就是两两进行比较,如果不满足次序就可以交换位置。比如,我们想要从小到大排序,通过两个位置上的值两两比较,如果逆序就交换,使关键字大的记录像泡泡一样冒出来放在末尾。

冒泡排序

冒泡排序是一种最简单的交换排序算法。


什么是交换?交换就是两两进行比较,如果不满足次序就可以交换位置。比如,我们想要从小到大排序,通过两个位置上的值两两比较,如果逆序就交换,使关键字大的记录像泡泡一样冒出来放在末尾。


重复执行若干次冒泡排序,最终得到有序序列。


冒泡排序的名字来源于:值较小的元素如同”气泡“一样逐渐漂浮到序列的顶端。


image.png

思想

  1. 给定一个 N 个元素的数组,冒泡法排序将:


如果元素大小关系不正确,交换这两个数(在本例中为 a > b),


  1. 比较一对相邻元素(a,b),
  2. 重复步骤 1 和 2,直到我们到达数组的末尾(最后一对是第(N-2)和(N-1)项,因为我们的数组从零开始)
  3. 到目前为止,最大的元素将在最后的位置。 然后我们将 N 减少 1,并重复步骤 1,直到 N = 1。

动画演示

给定一个数组 29, 10, 14, 37, 14, 48, 17 ,经过冒泡排序的动画如下所示:


image.png


代码实现

package main
import "fmt"
func main() {
    // index starts from 0
    var nums = []int{29, 10, 14, 37, 14, 48, 17}
    var length = len(nums)
    fmt.Println("原数组:", nums)
    bubbleSort(nums, length)
    fmt.Print("数组排序后:")
    for i := 0; i < length; i++ {
        fmt.Printf("%d\t", nums[i])
    }
}
func bubbleSort(nums []int, length int) {
    for i := 0; i < length-1; i++ {
        for j := 0; j < length-i-1; j++ {
            if nums[j] > nums[j+1] { // 如果大,交换
                var temp = nums[j] // 临时变量保存前一个值
                nums[j] = nums[j+1]
                nums[j+1] = temp
            }
        }
    }
}


运行上述代码,可以看到排序的结果:

[Running] go run "e:\Coding Workspaces\LearningGoTheEasiestWay\Go 数据结构\冒泡排序\main.go"
原数组: [29 10 14 37 14 48 17]
数组排序后:10    14  14  17  29  37  48

冒泡排序的优化一

可以附加一个标志来改进该算法,在排序过程中,如果没有交换操作意味着排序完成。如果序列已经是有序的,则可以通过判断该标记来结束算法。但优化后排序的时间复杂度没有发生量级的改变。

func bubbleSortImproved(nums []int, length int) {
    swapped := true
    for i := 0; i < length-1; i++ {
        for j := 0; j < length-i-1; j++ {
            if nums[j] > nums[j+1] {    // 如果大,交换
                var temp = nums[j]      // 临时变量保存前一个值
                nums[j] = nums[j+1]
                nums[j+1] = temp
                swapped = false         //如果没有出现这步骤,说明没有需要交换的了
            }
        }
        if swapped {    // 从头到尾都没有进行交换,说明已经排序完成
            break
        }
    }
}

在最好情况下,改进的冒泡算法的时间复杂度为 O(n)

冒泡排序优化二:记录发生交换的位置

其实冒泡算法还有第二种优化方式,如果有 10 个数的数组,仅前面 5 个无序,后面 5 个都已排好序且都大于前面 5 个数字,那么在第一趟遍历后,最后发生交换的位置必定小于 5 ,且这个位置之后的数据必定已经有序了。


改进思路:


记录某次遍历时最后发生数据交换的位置 lastSwap ,这个位置之后的数据显然已经有序了。因此通过记录最后发生数据交换的位置就可以确定下次循环的范围了。由于 lastSwap 位置之后的记录均已交换到位,故在进行下一趟排序时只要扫描到 lastSwap 位置即可。


读者可以动手试试跑一下如下的代码:

func BubbleBestSort(nums []int, length int) {
    lastSwap := length - 1
    lastSwapTemp := length - 1
    for j := 0; j < length; j++ {
        lastSwap = lastSwapTemp
        for i := 0; i < lastSwap; i++ {
            if nums[i] > nums[i+1] {
                nums[i], nums[i+1] = nums[i+1], nums[i]
                lastSwapTemp = i
            }
        }
        if lastSwap == lastSwapTemp {
            break
        }
    }
}

总结

冒泡排队相对比其他排序算法的优点是,它能够检测输入序列是否已经是排序的。但是数据量大的冒泡排序的性能就较差了,敬请期待其他的算法。


时间复杂度:冒泡排序的时间复杂度为 O(n2) (即使在最好的情况)


空间复杂度:空间复杂度为 O(1)

image.png


稳定性:冒泡排序同时是稳定的算法。

相关文章
|
3月前
|
消息中间件 缓存 NoSQL
Redis各类数据结构详细介绍及其在Go语言Gin框架下实践应用
这只是利用Go语言和Gin框架与Redis交互最基础部分展示;根据具体业务需求可能需要更复杂查询、事务处理或订阅发布功能实现更多高级特性应用场景。
293 86
|
5月前
|
搜索推荐 算法 Go
Go语言数组排序(冒泡排序法)—— 用最直观的方式掌握排序算法
本案例介绍使用冒泡排序对整数数组进行升序排序的实现方法,涵盖输入处理、错误检查与排序逻辑。通过代码演示和算法解析,帮助理解排序原理及Go语言切片操作,为学习更复杂排序算法打下基础。
|
5月前
|
存储 监控 算法
公司员工泄密防护体系中跳表数据结构及其 Go 语言算法的应用研究
在数字化办公中,企业面临员工泄密风险。本文探讨使用跳表(Skip List)数据结构优化泄密防护系统,提升敏感数据监测效率。跳表以其高效的动态数据处理能力,为企业信息安全管理提供了可靠技术支持。
133 0
|
存储 Go 容器
深入探究Go语言中的数据结构
深入探究Go语言中的数据结构
301 3
|
11月前
|
存储 安全 Go
Go语言中的map数据结构是如何实现的?
Go 语言中的 `map` 是基于哈希表实现的键值对数据结构,支持快速查找、插入和删除操作。其原理涉及哈希函数、桶(Bucket)、动态扩容和哈希冲突处理等关键机制,平均时间复杂度为 O(1)。为了确保线程安全,Go 提供了 `sync.Map` 类型,通过分段锁实现并发访问的安全性。示例代码展示了如何使用自定义结构体和切片模拟 `map` 功能,以及如何使用 `sync.Map` 进行线程安全的操作。
303 9
|
算法 搜索推荐
数据结构与算法学习十一:冒泡排序、选择排序、插入排序
本文介绍了冒泡排序、选择排序和插入排序三种基础排序算法的原理、实现代码和测试结果。
450 0
数据结构与算法学习十一:冒泡排序、选择排序、插入排序
|
搜索推荐
【数据结构常见七大排序(三)上】—交换排序篇【冒泡排序】And【快速排序】
【数据结构常见七大排序(三)上】—交换排序篇【冒泡排序】And【快速排序】
【数据结构常见七大排序(三)上】—交换排序篇【冒泡排序】And【快速排序】
|
搜索推荐 算法 Go
深入探索堆:Go语言中的高效数据结构
深入探索堆:Go语言中的高效数据结构
|
存储 算法 搜索推荐
【初阶数据结构篇】冒泡排序和快速排序(中篇)
与直接插入排序法相比,比较次数一致,但冒泡排序的交换需要执行三次,而直接插入排序因为使用了tmp临时变量存储要插入的数据,只用执行一次,所以直接插入排序法效率明显更高。
96 1
|
算法 Go
Go 语言 实现冒泡排序
冒泡排序是大家熟知的经典算法。在Go语言中实现它,关键在于理解其核心思想:通过不断比较并交换相邻元素,让序列中的最大值像泡泡一样“浮”至顶端。每一轮比较都能确定一个最大值的位置。外层循环控制排序轮数,内层循环负责比较与交换。随着每轮排序完成,未排序部分逐渐缩小,直至整个数组有序。以下是Go语言实现示例及说明。
254 1

热门文章

最新文章