Go归并排序算法的实现方法


Posted in Golang onApril 06, 2022

今天继续基础排序算法的图解和Go 代码实现,这次分享一个时间复杂度为*** 诶,时间复杂度多少先保密,文末会有分析。这次分享的排序算法是—归并排序(Merge Sort)。

归并排序的思想

与快速排序一样,归并排序采用的也是分治的策略,把原本的问题先分解成一些小问题进行求解,再把这些小问题各自的答案修整到一起得到原本问题的答案,从而达到分而治之的目的。

归并排序算法会把要排序的序列分成长度相当的两个子序列,当分无可分每个子序列中只有一个数据的时候,就对子序列进行归并。

归并指的是把两个排序好的子序列合并成一个有序序列。该操作会一直重复执行,直到所有子序列归并为一个整体为止。

归并排序的过程下面我们依然用图例过一遍归并排序对一个序列进行排序的过程。

图例出自—《我的第一本算法书》

首先,假设有下面这样一个待排序的序列:

Go归并排序算法的实现方法

待排序的一串数字

将序列以对半分割的形式分成两段 。

Go归并排序算法的实现方法

把序列二分成两段

再继续对子序列进行对半分割,分解下去 。

Go归并排序算法的实现方法

再继续往下分

直到分无可分,每个子序列中只有一个数据 。

Go归并排序算法的实现方法

分解到每个子序列只有一个数据

接下来对分割后的数据进行合并,合并时需要将数字按从小到大的顺序排列。

Go归并排序算法的实现方法

合并序列时按大小排序

把 6 和 4 合并,合并时按照数字大小排序,合并后的顺序为【4,6】,接下来把 3 和 7 合并,合并后的顺序为【3,7】。

[继续按照大小顺序合并后面的序列](/Users/klein/Library/Application Support/typora-user-images/image-20220405142734949.png)。

下面,我们看看怎么合并【4,6】和【3,7】这两个序列。合并这种含有多个数字的子序列时,要先比较首位数字,再移动较小的数字。

Go归并排序算法的实现方法

合并多元素的序列时,从首位开始比较,小的先移动

这里要比较两个子序列的首位数字是4 和 3。由于 4 > 3,所以合并序列时先移动 3。

Go归并排序算法的实现方法

4 > 3,所以合并序列时先移动 3

接下来,再按照比较两个序列首位,小的先合并,大的留下来继续比较的规则合并两个序列。

4 小于 7,所以先移动 4 到合并的序列。

Go归并排序算法的实现方法

由于4<7,所以移动4

两个子序列剩下的元素中,6 小于 7,所以先移动 6 。

Go归并排序算法的实现方法

6 < 7 所以先移动 6

最后移动剩下的 7。

Go归并排序算法的实现方法

子序列最后剩下了7,合并到序列中去

递归执行上面的操作,直到所有的数字都合并到一个整体的序列上为止。

Go归并排序算法的实现方法

小序列合并成两个大的序列

Go归并排序算法的实现方法

再继续往完整的序列上合并

最后得到一个完整的排序完成的序列 。

Go归并排序算法的实现方法

排序完成的序列

归并排序的 Go 代码实现

下面上一个用归并排序的Go代码实现,代码很简单,实现步骤就都放在了代码的注释里,就不再多说啦,先收藏文章(也要记得点赞),等有时间了自己在电脑上运行一下试试吧。

package main

import "fmt"

// 自顶向下归并排序,排序范围在 [begin,end) 的数组
func MergeSort(array []int, begin int, end int) {
    // 元素数量大于1时才进入递归
    if end - begin &gt; 1 {

        // 将数组一分为二,分为 array[begin,mid) 和 array[mid,high)
        mid := begin + (end-begin+1)/2

        // 先将左边排序好
        MergeSort(array, begin, mid)

        // 再将右边排序好
        MergeSort(array, mid, end)

        // 两个有序数组进行合并
        merge(array, begin, mid, end)
    }
}

// 归并操作
func merge(array []int, begin int, mid int, end int) {
    // 申请额外的空间来合并两个有序数组,这两个数组是 array[begin,mid),array[mid,end)
    leftSize := mid - begin         // 左边数组的长度
    rightSize := end - mid          // 右边数组的长度
    newSize := leftSize + rightSize // 辅助数组的长度
    result := make([]int, 0, newSize)

    l, r := 0, 0
    for l &lt; leftSize &amp;&amp; r &lt; rightSize {
        lValue := array[begin+l] // 左边数组的元素
        rValue := array[mid+r]   // 右边数组的元素
        // 小的元素先放进辅助数组里
        if lValue &lt; rValue {
            result = append(result, lValue)
            l++
        } else {
            result = append(result, rValue)
            r++
        }
    }

    // 将剩下的元素追加到辅助数组后面
    result = append(result, array[begin+l:mid]...)
    result = append(result, array[mid+r:end]...)

    // 将辅助数组的元素复制回原数组,这样该辅助空间就可以被释放掉
    for i := 0; i &lt; newSize; i++ {
        array[begin+i] = result[i]
    }
    return
}

归并排序的时间复杂度

老规矩,看完算法思想和实现步骤后,我们再来分析一下归并排序算法的时间复杂度。

归并排序中,分割序列所花费的时间不算在运行时间内 (可以当作序列本来就是分 割好的)。在合并两个已排好序的子序列时,只需依次比较处在序列首位数据的大小,然后移动较小的数据,因此只需花费和两个子序列的长度相应的运行时间。也就是说,完成一行归并所需的运行时间取决于这一行的数据量。

看一下这个图便能得知,无论哪一行都是 n 个数据,所以每行的运行时间都为 O(n)。

Go归并排序算法的实现方法

归并排序每一行的数据都是 n 个

而将长度为 n 的序列对半分割直到只有一个数据为止时,可以分成 行,因此,总共有 log2n 行。也就是说,总的运行时间为 ,这与前面讲到的快速排序相同。

到此这篇关于Go归并排序算法的实现方法的文章就介绍到这了,更多相关go归并排序算法内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Golang 相关文章推荐
解决Go gorm踩过的坑
Apr 30 Golang
解决go在函数退出后子协程的退出问题
Apr 30 Golang
K8s部署发布Golang应用程序的实现方法
Jul 16 Golang
Go 语言中 20 个占位符的整理
Oct 16 Golang
深入理解go缓存库freecache的使用
Feb 15 Golang
简单聊聊Golang中defer预计算参数
Mar 25 Golang
Go语言安装并操作redis的go-redis库
Apr 14 Golang
Go语言 详解net的tcp服务
Apr 14 Golang
Golang Elasticsearches 批量修改查询及发送MQ
Apr 19 Golang
Golang 遍历二叉树
Apr 19 Golang
Golang jwt身份认证
Apr 20 Golang
Go gorilla/sessions库安装使用
Aug 14 Golang
golang操作rocketmq的示例代码
Apr 06 #Golang
victoriaMetrics库布隆过滤器初始化及使用详解
如何解决goland,idea全局搜索快捷键失效问题
golang为什么要统一错误处理
简单聊聊Golang中defer预计算参数
Mar 25 #Golang
Go 中的空白标识符下划线
golang生成vcf通讯录格式文件详情
You might like
浅谈PHP发送HTTP请求的几种方式
2017/07/25 PHP
将光标定位于输入框最右侧实现代码
2012/12/04 Javascript
angularjs实现与服务器交互分享
2014/06/24 Javascript
js实现div弹出层的方法
2014/11/20 Javascript
Javascript中prototype属性实现给内置对象添加新的方法
2015/05/14 Javascript
nodejs实现获取当前url地址及url各种参数值
2015/06/25 NodeJs
在JavaScript应用中使用RequireJS来实现延迟加载
2015/07/01 Javascript
JS三级可折叠菜单实现方法
2016/02/29 Javascript
原生JS封装Ajax插件(同域、jsonp跨域)
2016/05/03 Javascript
使用Promise链式调用解决多个异步回调的问题
2017/01/15 Javascript
简单谈谈关于 npm 5.0 的新坑
2017/06/08 Javascript
浅谈在node.js进入文件目录的问题
2018/05/13 Javascript
centos 上快速搭建ghost博客方法分享
2018/05/23 Javascript
基于jQuery实现的设置文本区域的光标位置
2018/06/15 jQuery
Vue实现自定义下拉菜单功能
2018/07/16 Javascript
JavaScript常见事件处理程序实例总结
2019/01/05 Javascript
小程序:授权、登录、session_key、unionId的详解
2019/05/15 Javascript
基于js实现复制内容到操作系统粘贴板过程解析
2019/10/11 Javascript
jquery实现进度条状态展示
2020/03/26 jQuery
jQuery实现滑动开关效果
2020/08/02 jQuery
在Linux下使用Python的matplotlib绘制数据图的教程
2015/06/11 Python
Python3字符串学习教程
2015/08/20 Python
Python随机生成均匀分布在三角形内或者任意多边形内的点
2017/12/14 Python
Python 3.x基于Xml数据的Http请求方法
2018/12/28 Python
python实现小世界网络生成
2019/11/21 Python
python多线程实现同时执行两个while循环的操作
2020/05/02 Python
python 浮点数四舍五入需要注意的地方
2020/08/18 Python
python之随机数函数的实现示例
2020/12/30 Python
html5 Canvas画图教程(11)—使用lineTo/arc/bezierCurveTo画椭圆形
2013/01/09 HTML / CSS
雅诗兰黛旗下走天然植物路线的彩妆品牌:Prescriptives
2016/08/14 全球购物
Java面试题:Java类的Main方法如果是Private将会怎么样
2016/08/18 面试题
Java软件工程师综合面试题笔试题
2013/09/08 面试题
大学生就业推荐信范文
2013/11/29 职场文书
先进德育工作者事迹材料
2014/01/24 职场文书
商铺门面租房协议书
2014/10/21 职场文书
Django利用AJAX技术实现博文实时搜索
2021/05/06 Python