演算法 | 歸併排序

来源:https://www.cnblogs.com/yiluolion/archive/2020/04/30/12810702.html
-Advertisement-
Play Games

歸併排序 歸併排序演算法的核心就是 “歸併”,將兩個有序的數列合併,形成更大的有序數組。 歸併排序的原理 上面說了,歸併排序的核心就是“歸併”。如果排序一個數組,那麼將數組從中間分成前後兩部分,對前後兩部分分別進行排序,然後再將排序好的合併在一起,那麼這樣整個數組就會成為更大的有序數組。例如下麵示圖: ...


歸併排序


歸併排序演算法的核心就是 “歸併”,將兩個有序的數列合併,形成更大的有序數組。

歸併排序的原理


上面說了,歸併排序的核心就是“歸併”。如果排序一個數組,那麼將數組從中間分成前後兩部分,對前後兩部分分別進行排序,然後再將排序好的合併在一起,那麼這樣整個數組就會成為更大的有序數組。例如下麵示圖:

歸併排序分解圖例

歸併排序使用的思想是分治思想,即是分而治之。將複雜問題分解成兩個或者多個規模相同或類似的子問題,然後繼續細化,當子問題足夠簡單,能夠被求解,那麼複雜的問題也就能夠被求解出來。

這裡跟遞歸的思想很像。遞歸也是將大問題細化為小問題,找出終止條件和遞推公式。分治演算法一般都是用遞歸實現的。分治的思想是一種解決問題的處理思想,遞歸是一種編程技巧,兩者不會衝突。

上面說了,歸併排序能夠使用遞歸實現,那麼現在先寫出遞推公式和終止條件。

# 遞推公式
merge_sort(a...b) = merge(merge_sort(a...k), merge_sort(k+1, b))

# 終止條件
a >= b,不再繼續分解

這裡解釋下上面的公式,終止條件不細講,這裡表示的,索引 a 大於等於 索引 b,即是已經到達數組末尾,不能夠再細分。

講下遞推公式,merge_sort(a...b),表示給索引 a 到 k 之間的數組進行排序。這裡將分體轉為為兩個子問題,merge_sort(a...k)merge_sort(k+1...b),其中 k 表示原數組中間的位置。而 merge() 函數表示當兩個將兩個排好序的子數組合併,這樣就能夠使原數組實現排序。

現在將這個遞推公式轉化為代碼(這裡使用偽代碼):

# nums 表示數組,a 為起始點,b 為數組末尾索引
def merge_sort(nums, a, b):
    # 終止條件
    if a >= b:
        return
    
    # 取中間位置
    k = (a + b) // 2
    # 分治遞歸
    merge_sort(nums, a, k)
    merge_sort(nums, k+1, b)
    merge(nums[a...b], nums[a...k], nums[k+1, b])

最後 merge() 函數的作用,就是將後面兩個子數組合併好後放到 nums[a...b] 中。現在主要的問題是如何實現分解後排序合併?

先用一個簡單的示例,如下示圖:

merge

先申請臨時數組 temp,定義指針 i,j 分別指向 兩個子數組的第一個元素。比較 nums[i],nums[j],較小的元素放入臨時數組中,同時該元素的指針向右移動。

迴圈上面的過程,直到其中一個子數組的所有數據都放入臨時數組中,這個時候,將另外一個數組剩下的部分也放到臨時數組中。這個臨時數組就是最終合併的結果,將其複製回原數組即可。

現在用偽代碼實現這個過程:

def merge(nums[a...b], nums[a...k], nums[k+1, b]):
    # length 表示數組大小
    temp = [0] * length

    # 初始化指針,i, j, ti
    i, j, ti, = a, k+1, 0
    # 取小值放入臨時數組中,
    while (i <= k and j <= b):
        if (nums[i]<=nums[j]):
            temp[ti]=nums[i]
            i+=1
        else:
            temp[ti]=nums[j]
            j+=1
        ti+=1

    # 合併時會出現一個數組全部放入臨時數組中,
    # 另外一個數組還有剩餘,這裡將剩餘部分也放到臨時數組中
    if i < nums[a...k].length:
        for x in range(i, nums[a...k].length):
            temp[ti] = nums[x]
            ti += 1
    else:
        for x in range(j, nums[k+1...b].length):
            temp[ti] = nums[x]
            ti += 1

    # 如果需要複製回原數組,也可以直接複製
    return temp

以上本篇幅就是關於歸併排序的主要內容。


歡迎關註微信公眾號《書所集錄》


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • golang包管理 一、GOPATH GOPATH: 進行golang開發時的工作空間,你編寫的go源代碼和編譯後生成的可執行程式都將存放在GOPATH下。註意,GOPATH只是一個普通的文件目錄並且你所有的編碼工作都應該在該目錄下完成(golang 1.11版本引入 包依賴管理工具go mod,可 ...
  • 我的LeetCode:https://leetcode cn.com/u/ituring/ 我的LeetCode刷題源碼[GitHub]:https://github.com/izhoujie/Algorithmcii LeetCode 202. 快樂數 題目 編寫一個演算法來判斷一個數 n 是不是快 ...
  • [toc] C++11出現的右值相關語法可謂是很多C++程式員難以理解的新特性,不少人知其然而不知其所以然,面試被問到時大概就只知道可以減少開銷,但是為什麼減少開銷、減少了多少開銷、什麼時候用...這些問題也不一定知道,於是我寫下了這篇夾帶自己理解的博文,希望它對你有所幫助。 淺拷貝、深拷貝 在介紹 ...
  • Response對象 功能:設置響應消息 1. 設置響應行 1. 格式:HTTP/1.1 200 ok 2. 設置狀態碼:setStatus(int sc) 2. 設置響應頭:setHeader(String name,String value) 3. 設置響應體 使用步驟 1. 獲取輸出流 字元輸 ...
  • 什麼是函數重載?簡單的理解,支持多個同名函數的定義,只是參數的個數或者類型不同,在調用的時候,解釋器會根據參數的個數或者類型,調用相應的函數。 重載這個特性在很多語言中都有實現,比如 C++、Java 等,而 Python 並不支持。這篇文章呢,通過一些小技巧,可以讓 Python 支持類似的功能。 ...
  • 最近Switch上的《動物森友會》可謂是炙手可熱,它幾乎算是任天堂版的《模擬人生》了,它的最新游戲《集合啦!動物森友會》(以下稱“動森”)在發售後,取得了不錯的媒體評價和首發成績。 動森火起來有大部分原因是因為它的細節做的很到位,例如最受好評的:玩家可以自己手工DIY。(說實話,如果不是動森,我的N ...
  • Composer 使用不同的技術和標準簡化了類的自動載入。當今最常見的自動載入標準是 PSR-4: "autoload": { "psr-4": { "App\\": "src/" } } 這將使用帶有 “App” 名稱空間首碼的 PSR-4 標准將 src 文件夾中的所有類自動載入。但是,我們如何 ...
  • 前言 最近娛樂圈可以說得上是熱鬧非凡,前有霸道總裁愛小三,正宮撕逼網紅女,後有陽光大男孩羅志祥,被周揚青扒的名聲掃地。貴圈的愛情故事,常人是難以理解的,正如賈旭明張康這段相聲所說的這樣,娛樂圈的愛情總是分分合合,成為老百姓茶餘飯後的談資,城外的人想進去,城裡的人真會玩。 各種版本的洗白、謠言遍地亂飛 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...