<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
堆排序是一種樹形選擇排序演演算法。
簡單選擇排序演演算法每次選擇一個關鍵字最小的記錄需要 O(n) 的時間,而堆排序選擇一個關鍵字最小的記錄需要 O(nlogn)的時間。
堆可以看作一棵完全二元樹的順序儲存結構。
在這棵完全二元樹中,如果每個節點的值都大於等於左邊孩子的值,稱為大根堆(最大堆、又叫大頂堆)。如果每個節點的值都小於等於左邊孩子的值,稱為小根堆(最小堆,小頂堆)。
可以,用數學符號表示如下:
假如,{1, 7, 9, 2, 4, 6, 3, 5, 8}
建堆,然後進行堆排序輸出。
首先根據無序序列 {1, 7, 9, 2, 4, 6, 3, 5, 8}
按照完全二元樹的順序構建一棵完全二元樹,如圖:
然後從最後一個分支節點 n/2開始調整堆,這裡 9 / 2 = 4:
然後從 n/2−1 開始調整,即序號 3 開始調整,接著從 n/2-2 執行調整操作,如圖所示:
一直重複到序號為 1 的節點:
最終通過此次調整堆,得到新的堆為 [9, 8, 6, 7, 4, 1, 3, 5, 2]
,得到新的堆後開始堆排序過程
構建完初始堆後,此時,我們可以進入堆排序,從上面的方法中,
我們可以已知我們構建的最大堆的堆頂是最大的記錄,可以可以將堆頂交換到最後一個元素的位置,然後執行堆頂下沉操作,然後再執行堆調整操作(新的堆頂也是最大值),直到剩餘一個節點,得到一個有序序列。
此時,我們又可以進行堆調整操作,如下圖:
堆調整完畢,開始把新的堆頂 8 和最後一個記錄 2 進行交換,然後將堆頂下沉,調整為堆,如下圖所示:
從此我們得到新的堆頂 7 ,然後把 7 跟最後一個元素 3 進行交換,7 下沉,然後堆調整,慢慢得到堆頂 6 和 堆頂5,如圖所示:
然後是 3 下沉:
最後,堆頂 2 與最後一個記錄 1 進行交換,只剩一個節點,堆排序結束,如下圖所示:
我們得到的新的序列按序號讀取資料,就是一個有序序列。
最後,我們用程式碼來檢驗一下我們的動畫過程是否正確,如下:
package main import "fmt" // 調整堆 func adjustHeap(array []int, currentIndex int, maxLength int) { var noLeafValue = array[currentIndex] // 當前非葉子節點 // j 指向左孩子 // 當前非葉子節點的左節點為:2 * currentIndex + 1 for j := 2*currentIndex + 1; j <= maxLength; j = currentIndex*2 + 1 { if j < maxLength && array[j] < array[j+1] { // 如果有右孩子,且左孩子比右孩子小 j++ // j 指向右孩子 } if noLeafValue >= array[j] { break // 非葉子節點大於孩子節點,跳過不交換 } array[currentIndex] = array[j] // 移動到當前節點的父節點 currentIndex = j // j 指向交換後的新位置,繼續向下比較 } array[currentIndex] = noLeafValue // 放在合適的位置 } // 初始化堆 func createHeap(array []int, length int) { // 建堆 for i := length / 2; i >= 0; i-- { adjustHeap(array, i, length-1) } } func heapSort(array []int, length int) { for i := length - 1; i > 0; i-- { array[0], array[i] = array[i], array[0] adjustHeap(array, 0, i-1) } } func main() { var unsorted = []int{1, 7, 9, 2, 4, 6, 3, 5, 8} var length = len(unsorted) fmt.Println("建堆之前:") for i := 0; i < length; i++ { fmt.Printf("%d,", unsorted[i]) } fmt.Println() fmt.Println("建堆之後:") createHeap(unsorted, length) for i := 0; i < length; i++ { fmt.Printf("%d,", unsorted[i]) } fmt.Printf("n堆排序之後: n") heapSort(unsorted, length) for i := 0; i < length; i++ { fmt.Printf("%d,", unsorted[i]) } }
執行結果:
[Running] go run "e:Coding WorkspacesLearningGoTheEasiestWayGo 資料結構堆排序main.go"
建堆之前:
1,7,9,2,4,6,3,5,8,
建堆之後:
9,8,6,7,4,1,3,5,2,
堆排序之後:
1,2,3,4,5,6,7,8,9,
可以看到,建立堆的結果 9,8,6,7,4,1,3,5,2
和排序結果 1,2,3,4,5,6,7,8,9
都是和我們圖中的堆一樣,所以說圖看懂了程式碼也就變得有意思了。
總結一下堆排序的複雜度:
時間複雜度:堆排序主要耗費時間在初始堆和反覆調整堆上,所以時間複雜度為 O(nlogn)O(nlogn)O(nlogn)
空間複雜度:交換記錄需要一個輔助空間,所以空間複雜度為 O(1)O(1)O(1)
穩定性:堆排序多次交換關鍵字,可能會發生相等關鍵字排序前後位置不一樣的情況,所以不穩定
推薦大家都自己畫圖體驗一下堆排序的過程,這中間設計除了涉及到演演算法的精妙,也能體會到二元樹的遍歷過程。
以上就是Go 資料結構之堆排序範例詳解的詳細內容,更多關於Go 資料結構堆排序的資料請關注it145.com其它相關文章!
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45