再也不怕冒泡排序了,看完这篇Go语言详解就会了

简介: 再也不怕冒泡排序了,看完这篇Go语言详解就会了

/ Go 语言实现冒泡排序完全指南 /

本文将手把手教你使用 Go 语言实现冒泡排序,并提供完整的示例代码。

1

 

1. 冒泡排序原理

冒泡排序的基本思想是:

  1. 从左到右不断交换相邻逆序的元素
  2. 一轮下来,最大元素会交换移到右端
  3. 对除最后元素外的子序列,重复步骤 1
  4. 若没有发生交换,说明序列已排序完成

以数组 [5, 1, 4, 2, 3] 为例:

[5, 1, 4, 2, 3] -> [1, 5, 4, 2, 3]  
[1, 5, 4, 2, 3] -> [1, 4, 5, 2, 3]
[1, 4, 5, 2, 3] -> [1, 4, 2, 5, 3]
[1, 4, 2, 5, 3] -> [1, 4, 2, 3, 5]

重复轮数直到完全有序。

2

 

2. 基本实现

冒泡排序标准实现:

func bubbleSort(nums []int) {
  for i := 0; i < len(nums); i++ {
    for j := 1; j < len(nums)-i; j++ {
      if nums[j] < nums[j-1] {
        nums[j], nums[j-1] = nums[j-1], nums[j]  
      }
    }
  }
}

内层循环处理交换,外层循环控制轮数。

3

 

3. 优化方法

设置标志提前退出

func bubbleSort(nums []int) {
  for i := 0; i < len(nums); i++ {
    swapped := false
    for j := 1; j < len(nums)-i; j++ {
      if nums[j] < nums[j-1] {
        nums[j], nums[j-1] = nums[j-1], nums[j]
        swapped = true
      }
    }
    if !swapped {
      break
    }
  }
}

如果没有交换发生则提前退出。

记录最后交换位置

func bubbleSort(nums []int) {
  for i := 0; i < len(nums); i++ {
    lastSwapIndex := 0
    for j := 1; j < len(nums)-i; j++ {
      if nums[j] < nums[j-1] {
        nums[j], nums[j-1] = nums[j-1], nums[j]
        lastSwapIndex = j 
      }
    }
    if lastSwapIndex == 0 {
      break
    }
  }
}

记录最后一次交换位置,次后仅需比较到这里。

4

 

4. 改进方法

鸡尾酒排序

func cocktailSort(nums []int) []int {
  for i := 0; i < len(nums)/2; i++ {
    nums = bubbleSortForward(nums)
    nums = bubbleSortBackward(nums)
  }
  return nums
}
// 正向冒泡
func bubbleSortForward(nums []int) []int {
  for i := 0; i < len(nums); i++ {
    for j := 1; j < len(nums)-i; j++ {
      if nums[j] < nums[j-1] {
        nums[j], nums[j-1] = nums[j-1], nums[j]
      }
    }
  }
  return nums
}
// 反向冒泡 
func bubbleSortBackward(nums []int) []int {
  n := len(nums)
  for i := 0; i < n; i++ {
    for j := n - 1; j > i; j-- {
      if nums[j] < nums[j-1] {
        nums[j], nums[j-1] = nums[j-1], nums[j]  
      }
    }
  }
  return nums
}

正反向交替冒泡,效率提升。

插入排序改进

func insertionBubbleSort(nums []int) {
  for i := 1; i < len(nums); i++ {
    insertIntoOrdered(nums, i)
  }
}
// 将nums[i]插入有序区间[0...i-1]
func insertIntoOrdered(nums []int, i int) {
  for j := i - 1; j >= 0 && nums[j] > nums[j+1]; j-- {
    nums[j], nums[j+1] = nums[j+1], nums[j] 
  }
}

    5

     

    5. 复杂度分析

    • 最优时间复杂度 O(n)
    • 最坏时间复杂度 O(n^2)
    • 平均时间复杂度 O(n^2)
    • 空间复杂度 O(1)

    冒泡排序是一种稳定的排序算法。

    6

     

    6. 拓展与优化

    并行冒泡

    func parallelBubbleSort(nums []int) {
      var wg sync.WaitGroup
      for i := 0; i < len(nums); i++ {
        wg.Add(1)
        go bubblePass(&wg, nums, i)
      }
      wg.Wait()
    }
    func bubblePass(wg *sync.WaitGroup, nums []int, i int) {
      defer wg.Done()
      for j := 1; j < len(nums)-i; j++ {
        if nums[j] < nums[j-1] {
          nums[j], nums[j-1] = nums[j-1], nums[j]
        }
      }
    }

    并发进行多轮冒泡,效率提升。

    可视化

    import "github.com/faiface/pixel"
    func visualizeBubbleSort(nums []int) {
      for i := 0; i < len(nums); i++ {
        for j := 0; j < len(nums)-i-1; j++ {
          if nums[j] > nums[j+1] {
            // 交换两个元素
            temp := nums[j]
            nums[j] = nums[j+1]
            nums[j+1] = temp
            // 绘制当前排序状态
            drawBars(nums) 
          }
        }
      }
    }
    func drawBars(nums []int) {
      canvas := pixel.NewCanvas(500, 500) 
      barWidth := canvas.Bounds().W / float64(len(nums))
      for i, v := range nums {
        x := float64(i) * barWidth
        h := float64(v) * 4 
        canvas.Rectangle(x, 500-h, barWidth, h) 
      }
      canvas.Draw()
      // 暂停一小段时间
      time.Sleep(time.Millisecond * 500)
    }

    通过动画等可视化冒泡排序过程。

    7

     

    总结

    这篇文章详细介绍了冒泡排序的原理、实现、分析和优化方法,每部分都以通俗易懂的语言和充足的代码进行讲解,全面帮助读者理解冒泡排序算法的方方面面。冒泡排序虽简单但启发意义深远,希望本文可以增进读者对算法和编码的理解。



    目录
    相关文章
    |
    2天前
    |
    安全 Java Go
    探索Go语言在高并发环境中的优势
    在当今的技术环境中,高并发处理能力成为评估编程语言性能的关键因素之一。Go语言(Golang),作为Google开发的一种编程语言,以其独特的并发处理模型和高效的性能赢得了广泛关注。本文将深入探讨Go语言在高并发环境中的优势,尤其是其goroutine和channel机制如何简化并发编程,提升系统的响应速度和稳定性。通过具体的案例分析和性能对比,本文揭示了Go语言在实际应用中的高效性,并为开发者在选择合适技术栈时提供参考。
    |
    6天前
    |
    运维 Kubernetes Go
    "解锁K8s二开新姿势!client-go:你不可不知的Go语言神器,让Kubernetes集群管理如虎添翼,秒变运维大神!"
    【8月更文挑战第14天】随着云原生技术的发展,Kubernetes (K8s) 成为容器编排的首选。client-go作为K8s的官方Go语言客户端库,通过封装RESTful API,使开发者能便捷地管理集群资源,如Pods和服务。本文介绍client-go基本概念、使用方法及自定义操作。涵盖ClientSet、DynamicClient等客户端实现,以及lister、informer等组件,通过示例展示如何列出集群中的所有Pods。client-go的强大功能助力高效开发和运维。
    27 1
    |
    6天前
    |
    SQL 关系型数据库 MySQL
    Go语言中使用 sqlx 来操作 MySQL
    Go语言因其高效的性能和简洁的语法而受到开发者们的欢迎。在开发过程中,数据库操作不可或缺。虽然Go的标准库提供了`database/sql`包支持数据库操作,但使用起来稍显复杂。为此,`sqlx`应运而生,作为`database/sql`的扩展库,它简化了许多常见的数据库任务。本文介绍如何使用`sqlx`包操作MySQL数据库,包括安装所需的包、连接数据库、创建表、插入/查询/更新/删除数据等操作,并展示了如何利用命名参数来进一步简化代码。通过`sqlx`,开发者可以更加高效且简洁地完成数据库交互任务。
    15 1
    |
    6天前
    |
    算法 NoSQL 中间件
    go语言后端开发学习(六) ——基于雪花算法生成用户ID
    本文介绍了分布式ID生成中的Snowflake(雪花)算法。为解决用户ID安全性与唯一性问题,Snowflake算法生成的ID具备全局唯一性、递增性、高可用性和高性能性等特点。64位ID由符号位(固定为0)、41位时间戳、10位标识位(含数据中心与机器ID)及12位序列号组成。面对ID重复风险,可通过预分配、动态或统一分配标识位解决。Go语言实现示例展示了如何使用第三方包`sonyflake`生成ID,确保不同节点产生的ID始终唯一。
    go语言后端开发学习(六) ——基于雪花算法生成用户ID
    |
    7天前
    |
    JSON 缓存 监控
    go语言后端开发学习(五)——如何在项目中使用Viper来配置环境
    Viper 是一个强大的 Go 语言配置管理库,适用于各类应用,包括 Twelve-Factor Apps。相比仅支持 `.ini` 格式的 `go-ini`,Viper 支持更多配置格式如 JSON、TOML、YAML
    go语言后端开发学习(五)——如何在项目中使用Viper来配置环境
    |
    1天前
    |
    监控 NoSQL Go
    Go语言中高效使用Redis的Pipeline
    Redis 是构建高性能应用时常用的内存数据库,通过其 Pipeline 和 Watch 机制可批量执行命令并确保数据安全性。Pipeline 类似于超市购物一次性结账,减少网络交互时间,提升效率。Go 语言示例展示了如何使用 Pipeline 和 Pipelined 方法简化代码,并通过 TxPipeline 保证操作原子性。Watch 机制则通过监控键变化实现乐观锁,防止并发问题导致的数据不一致。这些机制简化了开发流程,提高了应用程序的性能和可靠性。
    6 0
    |
    4天前
    |
    NoSQL Go Redis
    Go语言中如何扫描Redis中大量的key
    在Redis中,遍历大量键时直接使用`KEYS`命令会导致性能瓶颈,因为它会一次性返回所有匹配的键,可能阻塞Redis并影响服务稳定性。为解决此问题,Redis提供了`SCAN`命令来分批迭代键,避免一次性加载过多数据。本文通过两个Go语言示例演示如何使用`SCAN`命令:第一个示例展示了基本的手动迭代方式;第二个示例则利用`Iterator`简化迭代过程。这两种方法均有效地避免了`KEYS`命令的性能问题,并提高了遍历Redis键的效率。
    15 0
    |
    5天前
    |
    监控 Serverless Go
    Golang 开发函数计算问题之Go 语言中切片扩容时需要拷贝原数组中的数据如何解决
    Golang 开发函数计算问题之Go 语言中切片扩容时需要拷贝原数组中的数据如何解决
    |
    5天前
    |
    关系型数据库 MySQL 数据库连接
    Go语言中使用sqlx来操作事务
    在应用中,数据库事务保证操作的ACID特性至关重要。`github.com/jmoiron/sqlx`简化了数据库操作。首先安装SQLX和MySQL驱动:`go get github.com/jmoiron/sqlx`和`go get github.com/go-sql-driver/mysql`。导入所需的包后,创建数据库连接并使用`Beginx()`方法开始事务。通过`tx.Commit()`提交或`tx.Rollback()`回滚事务以确保数据一致性和完整性。
    8 0
    |
    7天前
    |
    SQL 安全 关系型数据库
    Go 语言中的 MySQL 事务操作
    在现代应用中,确保数据完整与一致至关重要。MySQL的事务机制提供了可靠保障。本文首先解释了事务的概念及其ACID特性,随后介绍了如何在Go语言中使用`database/sql`包进行MySQL事务操作。通过一个银行转账的例子,演示了如何通过Go开启事务、执行操作并在必要时回滚或提交,确保数据一致性。最后,还讨论了不同事务隔离级别的含义及如何在Go中设置这些级别。通过本文的学习,开发者能更好地掌握MySQL事务的应用。
    12 0