溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Go數(shù)據(jù)結構之堆排序怎么實現(xiàn)

發(fā)布時間:2022-08-27 10:58:57 來源:億速云 閱讀:119 作者:iii 欄目:開發(fā)技術

這篇“Go數(shù)據(jù)結構之堆排序怎么實現(xiàn)”文章的知識點大部分人都不太理解,所以小編給大家總結了以下內容,內容詳細,步驟清晰,具有一定的借鑒價值,希望大家閱讀完這篇文章能有所收獲,下面我們一起來看看這篇“Go數(shù)據(jù)結構之堆排序怎么實現(xiàn)”文章吧。

    堆排序

    堆排序是一種樹形選擇排序算法。

    簡單選擇排序算法每次選擇一個關鍵字最小的記錄需要 O(n) 的時間,而堆排序選擇一個關鍵字最小的記錄需要 O(nlogn)的時間。

    堆可以看作一棵完全二叉樹的順序存儲結構。

    在這棵完全二叉樹中,如果每個節(jié)點的值都大于等于左邊孩子的值,稱為大根堆(最大堆、又叫大頂堆)。如果每個節(jié)點的值都小于等于左邊孩子的值,稱為小根堆(最小堆,小頂堆)。

    可以,用數(shù)學符號表示如下:

    Go數(shù)據(jù)結構之堆排序怎么實現(xiàn)

    堆排序過程

    • 構建初始堆

    • 在輸出堆的頂層元素后,從上到下進行調整,將頂層元素與其左右子樹的根節(jié)點進行比較,并將最小的元素交換到堆的頂部;然后不斷調整直到葉子節(jié)點得到新的堆。

    假如,{1, 7, 9, 2, 4, 6, 3, 5, 8} 建堆,然后進行堆排序輸出。

    動畫顯示

    • 初始化堆,建堆操作圖畫演示:

    首先根據(jù)無序序列 {1, 7, 9, 2, 4, 6, 3, 5, 8} 按照完全二叉樹的順序構建一棵完全二叉樹,如圖:

    Go數(shù)據(jù)結構之堆排序怎么實現(xiàn)

    然后從最后一個分支節(jié)點 n/2開始調整堆,這里 9 / 2 = 4:

    Go數(shù)據(jù)結構之堆排序怎么實現(xiàn)

    然后從 n/2−1 開始調整,即序號 3 開始調整,接著從 n/2-2 執(zhí)行調整操作,如圖所示:

    一直重復到序號為 1 的節(jié)點:

    Go數(shù)據(jù)結構之堆排序怎么實現(xiàn)

    最終通過此次調整堆,得到新的堆為 [9, 8, 6, 7, 4, 1, 3, 5, 2] ,得到新的堆后開始堆排序過程

    開始堆排序

    構建完初始堆后,此時,我們可以進入堆排序,從上面的方法中,

    我們可以已知我們構建的最大堆的堆頂是最大的記錄,可以可以將堆頂交換到最后一個元素的位置,然后執(zhí)行堆頂下沉操作,然后再執(zhí)行堆調整操作(新的堆頂也是最大值),直到剩余一個節(jié)點,得到一個有序序列。

    Go數(shù)據(jù)結構之堆排序怎么實現(xiàn)

    此時,我們又可以進行堆調整操作,如下圖:

    Go數(shù)據(jù)結構之堆排序怎么實現(xiàn)

    堆調整完畢,開始把新的堆頂 8 和最后一個記錄 2 進行交換,然后將堆頂下沉,調整為堆,如下圖所示:

    Go數(shù)據(jù)結構之堆排序怎么實現(xiàn)

    從此我們得到新的堆頂 7 ,然后把 7 跟最后一個元素 3 進行交換,7 下沉,然后堆調整,慢慢得到堆頂 6 和 堆頂5,如圖所示:

    Go數(shù)據(jù)結構之堆排序怎么實現(xiàn)

    然后是 3 下沉:

    Go數(shù)據(jù)結構之堆排序怎么實現(xiàn)

    最后,堆頂 2 與最后一個記錄 1 進行交換,只剩一個節(jié)點,堆排序結束,如下圖所示:

    Go數(shù)據(jù)結構之堆排序怎么實現(xiàn)

    我們得到的新的序列按序號讀取數(shù)據(jù),就是一個有序序列。

    代碼實現(xiàn)

    最后,我們用代碼來檢驗一下我們的動畫過程是否正確,如下:

    package main
    import "fmt"
    // 調整堆
    func adjustHeap(array []int, currentIndex int, maxLength int) {
        var noLeafValue = array[currentIndex] // 當前非葉子節(jié)點
        // j 指向左孩子
        // 當前非葉子節(jié)點的左節(jié)點為: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 // 非葉子節(jié)點大于孩子節(jié)點,跳過不交換
            }
            array[currentIndex] = array[j] // 移動到當前節(jié)點的父節(jié)點
            currentIndex = j               // j 指向交換后的新位置,繼續(xù)向下比較
        }
        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 Workspaces\LearningGoTheEasiestWay\Go 數(shù)據(jù)結構\堆排序\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,

    可以看到,創(chuàng)建堆的結果 9,8,6,7,4,1,3,5,2 和排序結果 1,2,3,4,5,6,7,8,9 都是和我們圖中的堆一樣,所以說圖看懂了代碼也就變得有意思了。

    以上就是關于“Go數(shù)據(jù)結構之堆排序怎么實現(xiàn)”這篇文章的內容,相信大家都有了一定的了解,希望小編分享的內容對大家有幫助,若想了解更多相關的知識內容,請關注億速云行業(yè)資訊頻道。

    向AI問一下細節(jié)

    免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據(jù),一經查實,將立刻刪除涉嫌侵權內容。

    go
    AI