热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Go语言实现堆排序的详细教程

本文主要介绍了Go语言实现堆排序的详细教程,包括大根堆的定义和完全二叉树的概念。通过图解和算法描述,详细介绍了堆排序的实现过程。堆排序是一种效率很高的排序算法,时间复杂度为O(nlgn)。阅读本文大约需要15分钟。

篇首语:本文由编程笔记#小编为大家整理,主要介绍了Go(08)实现堆排序相关的知识,希望对你有一定的参考价值。



当前浏览器不支持播放音乐或语音,请在微信或其他浏览器中播放 Go(08)实现堆排序

阅读本文大约15分钟

本文难度:

前文介绍了golang实现基本的四中排序,本文带领大家实现堆排序,堆排序是效率很高的算法,通过取出大根堆堆顶元素从而实现排序的算法。
该算法以出色的效率著称,时间复杂度为O (nlgn)


堆排序描述


什么是大根堆

1 大根堆是一颗完全二叉树
2 该完全二叉树,根节点一定大于等于其左右子节点,并且大于等于其子树所有节点。


完全二叉树

完全二叉树是相对于满二叉树来讲的,对于深度为K的,有n个结点的二叉树,当且仅当其每一个结点都与深度为K的满二叉树中编号从1至n的结点一一对应时称之为完全二叉树。


满二叉树:

一棵深度为 k,且有 2k - 1 个节点称之为满二叉树,即每一层上的节点数都是最大节点数。

也就是说完全二叉树可以这么理解,在一棵二叉树中,除最后一层外,若其余层都是满的,并且最后一层或者是满的,或者是在右边缺少连续若干节点,则此二叉树为完全二叉树(Complete Binary Tree)

下图就是一个完全二叉树
Go(08)实现堆排序


算法图解

假设有序列
下标 | 0 | 1 | 2 | 3 | 4 | 5 |
数值 | 6 | 1 | 0 | 5 | 2 | 9 |
将该序列构造成上图的二叉树,首先从最后一个非叶子结点(也就是下标2的节点元素0)开始,将其所在子树调节为大根堆
Go(08)实现堆排序

由于9比0大,所以二者交换位置,这样下标2所在子树就变为大根堆了。接下来从右往左,从下往上依次处理所有非叶子节点所在子树,使其依次形成大根堆。下面处理下标为1的子树。
Go(08)实现堆排序

下标1的两个子节点中5比2大,选择最大的子节点5和1比较,由于5比1大,所以二者交换位置,这样下标1所在子树就变为大根堆了。接下来处理下标0所在子树,使其成为大根堆。

Go(08)实现堆排序

同样选择下标0最大的子节点9和下标0的元素6交换位置,这样最大元素9放在下标0的位置,由于6和9交换位置,还要考虑元素6所在的下标为1的子树是否因为交换导致失去大根堆特性,如果6比其子节点小,则继续将6下移,直到找到合理位置。此时6比其子节点0大,所以不需要移动了。

下标0就是根节点,到此为止一颗大根堆构造完成。
接下来将根节点9和最后一个元素交换,n为最后一个元素。这样调节前n-1节点,是这些节点构成大根堆。
具体操作如下图
交换最后一个元素和根节点
Go(08)实现堆排序

将最后一个元素排出不再比较,比较前n-1个元素,重新构造大根堆。
Go(08)实现堆排序

以此类推,
将n-1个节点调整为大根堆
Go(08)实现堆排序

前n-1调节为大根堆后,将根元素和第n-1元素交换

将第n-1个元素取出。

这样最后两个元素分别为6,9,是从小到大拍好的顺序,继续使前n-2节点形成大根堆。直到n=1


算法实现

首先定义一个HeapSort类,然后设计一个成员函数adjustHeap,该函数主要实现将index所在位置的子树构造成大根堆。
adjustHeap的三个参数分别为一段连续的数据序列,index表示子树根所在位置,length表示要排序的长度。










1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36

type HeapSort struct {
}
//调整index为根的子树,此时index的左右子树都是大根树
//比较index和其左右节点,将index根节点设置为最大的元素
//可能会引起子树失效,所以会循环处理修改的子树
func (hs *HeapSort) adjustHeap(array []int, index, length int) {
//index 的左右子节点
leftchild := index*2 + 1
rightchild := leftchild + 1
maxchild := leftchild
for {
//如果左节点比长度大,说明该节点为子节点
if leftchild > length-1 {
break
}
//右节点存在,且比左节点大
if rightchild <= length-1 && array[rightchild] > array[maxchild] {
maxchild = rightchild
}
//index 元素比最大子节点大,则不需要交换,退出
if array[index] > array[maxchild] {
break
}
//比较自己元素和最大节点的元素,做交换
hs.swap(array, index, maxchild)
index = maxchild
leftchild = index*2 + 1
rightchild = leftchild + 1
maxchild = leftchild
}
}



接下来实现一个小函数,用来交换两个位置的元素










1
2
3
4
5
6
7
8
9
10

func (hs *HeapSort) swap(array []int, i, j int) {
if i >= len(array) || j >= len(array) {
return
}
temp := array[i]
array[i] = array[j]
array[j] = temp
}



接下来实现将n个元素排序成大根堆,并且交换根元素和第n个元素,并将第n各元素排出,继续比较n-1个元素构造大根堆的逻辑










1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18

func (hs *HeapSort) sort(array []int) {
//每次循环后长度减少,因为每次循环最后元素都变为最大
for length := len(array); length > 1; length-- {
//最后一个非叶子节点索引
lastnode := length/2 - 1
//从最后一个非叶子节点一次从左到右,从下到上排序子树
//循环过后得到一个大顶堆(此时还不是大根堆)
for i := lastnode; i >= 0; i-- {
hs.adjustHeap(array, i, length)
}
//将堆顶元素放到末尾
hs.swap(array, 0, length-1)
fmt.Println(array)
}
}



ok,算法完成,可以进行测试了










1
2
3
4
5
6

func main() {
//数组初始化
array := []int{6, 1, 0, 5, 2, 9, 6}
hs := new(HeapSort)
hs.sort(array)
}



结果如下,打印的是每次排出元素后序列的结果










1
2
3
4
5
6

[6 5 6 1 2 0 9]
[0 5 6 1 2 6 9]
[2 5 0 1 6 6 9]
[1 2 0 5 6 6 9]
[0 1 2 5 6 6 9]
[0 1 2 5 6 6 9]



目前为止,堆排序介绍完了,欢迎下载源码

源码在阅读原文中


推荐阅读
  • 本文介绍了使用Java实现大数乘法的分治算法,包括输入数据的处理、普通大数乘法的结果和Karatsuba大数乘法的结果。通过改变long类型可以适应不同范围的大数乘法计算。 ... [详细]
  • LeetCode笔记:剑指Offer 41. 数据流中的中位数(Java、堆、优先队列、知识点)
    本文介绍了LeetCode剑指Offer 41题的解题思路和代码实现,主要涉及了Java中的优先队列和堆排序的知识点。优先队列是Queue接口的实现,可以对其中的元素进行排序,采用小顶堆的方式进行排序。本文还介绍了Java中queue的offer、poll、add、remove、element、peek等方法的区别和用法。 ... [详细]
  • C# 7.0 新特性:基于Tuple的“多”返回值方法
    本文介绍了C# 7.0中基于Tuple的“多”返回值方法的使用。通过对C# 6.0及更早版本的做法进行回顾,提出了问题:如何使一个方法可返回多个返回值。然后详细介绍了C# 7.0中使用Tuple的写法,并给出了示例代码。最后,总结了该新特性的优点。 ... [详细]
  • 本文详细介绍了Java中vector的使用方法和相关知识,包括vector类的功能、构造方法和使用注意事项。通过使用vector类,可以方便地实现动态数组的功能,并且可以随意插入不同类型的对象,进行查找、插入和删除操作。这篇文章对于需要频繁进行查找、插入和删除操作的情况下,使用vector类是一个很好的选择。 ... [详细]
  • 本文讨论了clone的fork与pthread_create创建线程的不同之处。进程是一个指令执行流及其执行环境,其执行环境是一个系统资源的集合。在调用系统调用fork创建一个进程时,子进程只是完全复制父进程的资源,这样得到的子进程独立于父进程,具有良好的并发性。但是二者之间的通讯需要通过专门的通讯机制,另外通过fork创建子进程系统开销很大。因此,在某些情况下,使用clone或pthread_create创建线程可能更加高效。 ... [详细]
  • 先看官方文档TheJavaTutorialshavebeenwrittenforJDK8.Examplesandpracticesdescribedinthispagedontta ... [详细]
  • 本文介绍了使用哈夫曼树实现文件压缩和解压的方法。首先对数据结构课程设计中的代码进行了分析,包括使用时间调用、常量定义和统计文件中各个字符时相关的结构体。然后讨论了哈夫曼树的实现原理和算法。最后介绍了文件压缩和解压的具体步骤,包括字符统计、构建哈夫曼树、生成编码表、编码和解码过程。通过实例演示了文件压缩和解压的效果。本文的内容对于理解哈夫曼树的实现原理和应用具有一定的参考价值。 ... [详细]
  • 深入解析Linux下的I/O多路转接epoll技术
    本文深入解析了Linux下的I/O多路转接epoll技术,介绍了select和poll函数的问题,以及epoll函数的设计和优点。同时讲解了epoll函数的使用方法,包括epoll_create和epoll_ctl两个系统调用。 ... [详细]
  • Java编程实现邻接矩阵表示稠密图的方法及实现类介绍
    本文介绍了Java编程如何实现邻接矩阵表示稠密图的方法,通过一个名为AMWGraph.java的类来构造邻接矩阵表示的图,并提供了插入结点、插入边、获取邻接结点等功能。通过使用二维数组来表示结点之间的关系,并通过元素的值来表示权值的大小,实现了稠密图的表示和操作。对于对稠密图的表示和操作感兴趣的读者可以参考本文。 ... [详细]
  • JVM:33 如何查看JVM的Full GC日志
    1.示例代码packagecom.webcode;publicclassDemo4{publicstaticvoidmain(String[]args){byte[]arr ... [详细]
  • 初识java关于JDK、JRE、JVM 了解一下 ... [详细]
  • 本文介绍了为什么要使用多进程处理TCP服务端,多进程的好处包括可靠性高和处理大量数据时速度快。然而,多进程不能共享进程空间,因此有一些变量不能共享。文章还提供了使用多进程实现TCP服务端的代码,并对代码进行了详细注释。 ... [详细]
  • 本文介绍了解决二叉树层序创建问题的方法。通过使用队列结构体和二叉树结构体,实现了入队和出队操作,并提供了判断队列是否为空的函数。详细介绍了解决该问题的步骤和流程。 ... [详细]
  • 李逍遥寻找仙药的迷阵之旅
    本文讲述了少年李逍遥为了救治婶婶的病情,前往仙灵岛寻找仙药的故事。他需要穿越一个由M×N个方格组成的迷阵,有些方格内有怪物,有些方格是安全的。李逍遥需要避开有怪物的方格,并经过最少的方格,找到仙药。在寻找的过程中,他还会遇到神秘人物。本文提供了一个迷阵样例及李逍遥找到仙药的路线。 ... [详细]
  • 如何在跨函数中使用内存?
    本文介绍了在跨函数中使用内存的方法,包括使用指针变量、动态分配内存和静态分配内存的区别。通过示例代码说明了如何正确地在不同函数中使用内存,并提醒程序员在使用动态分配内存时要手动释放内存,以防止内存泄漏。 ... [详细]
author-avatar
xinweiss
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有