演算法實例-C#-歸併排序-MergeSort

来源:http://www.cnblogs.com/aaron-clark-aic/archive/2016/04/06/5359676.html
-Advertisement-
Play Games

演算法實例 排序演算法Sort 歸併排序MergeSort 演算法說明 1. 歸併的思路是任意兩個元素可以比較大小,那麼任意兩個有序的元素集合也可以通過比較大小的方式歸併成一個有序的元素集合 2. 任何的無序元素集合可以拆分的最小的可比較單位是元素本身,例如List list = new List() { ...


# 演算法實例 #

排序演算法Sort

歸併排序MergeSort

演算法說明

  1. 歸併的思路是任意兩個元素可以比較大小,那麼任意兩個有序的元素集合也可以通過比較大小的方式歸併成一個有序的元素集合
  2. 任何的無序元素集合可以拆分的最小的可比較單位是元素本身,例如List list = new List() { 23, 42, 4, 16, 8, 23, 15, 3, 9, 55, 0, 23, 34, 12, 2, 46, 25, 25 };可以拆分到最小的元素為{{23},{42},{4},{16},{8},{23},{15},{3},{9},{55},{0},{23},{34},{12},{2},{46},{25},{25}},如果我們將這些元素組合兩兩比較就可以得到新的一組有序的元素組合{{23,24},{4,16},{8,23},{3,15},{9,55},{0,23},{12,34},{2,46},{25,25}}
  3. 對於上訴的元素集合進行比較後的有序元素集合,如下圖

最基本元素(即不可拆分最小元素集合)
最基本可比較元素
第一次兩兩歸併
第一次兩兩歸併
第二次兩兩歸併
第二次兩兩歸併
第三次兩兩歸併
第三次兩兩歸併
第四次兩兩歸併
第四次兩兩歸併
最終合一的歸併
最終合一的歸併


演算法步驟

1.我的演算法中使用隊列來作為容器容納元素,最小可比較單位也就是隊列中只有壹個元素,同時在外層也使用隊列來分別容納歸併的隊列即使用 Queue<Queue<T>> 的結構,其中 Queue<Queue<T>>是壹個歸併隊列的集合(如[{23,24},{4,16}]),Queue<T>是有序元素的集合(如{23,24})
2.遞歸的跳出條件是,歸併完成即Queue<Queue<T>>的元素個數Count為壹.且每一次遞歸完成一次兩兩歸併.
3.使用隊列的好處在於,可以使用Peek()和Dequeue()方法來確定參與一次歸併的隊列Queue<T> _tempLeft和Queue<T> _tempRight哪一個先出對歸並到新的有序集合中.

代碼示例

public static class MergeSorterA
{

    /// <summary>
    /// Public merge-sort API
    /// </summary>
    /// <typeparam name="T"></typeparam>
    /// <param name="collection"></param>
    /// <param name="comparer"></param>
    public static void MergeSortA<T>(this Queue<T> collection, Comparer<T> comparer = null)
    {
        comparer = comparer ?? Comparer<T>.Default;
        Queue<Queue<T>> _queue = new Queue<Queue<T>>();
        while (collection.Count !=0 )
        {
            Queue<T> _temp = new Queue<T>();
            _temp.Enqueue(collection.Dequeue());
            _queue.Enqueue(_temp);
        }

        _queue.InternalMergeSortA(comparer);

        foreach (var item in _queue.Dequeue())
        {
            collection.Enqueue(item);
        }
    }
    

    /// <summary>
    /// Implements the recursive merge-sort algorithm
    /// </summary>
    /// <typeparam name="T"></typeparam>
    /// <param name="collectionQueue"></param>
    /// <param name="comparer"></param>
    private static void InternalMergeSortA<T>(this Queue<Queue<T>> collectionQueue, Comparer<T> comparer)
    {
        Queue<Queue<T>> _listQueue = new Queue<Queue<T>>();
        
        while (collectionQueue.Count != 0)
        {
            Queue<T> _queue = new Queue<T>();
            Queue<T> _tempLeft = new Queue<T>();
            Queue<T> _tempRight = new Queue<T>();
            _tempLeft = collectionQueue.Dequeue();
            //make sure _tmepLeft is not the ends
            if (collectionQueue.Count != 0)
            {
                _tempRight = collectionQueue.Dequeue();
            }
            int _length = _tempLeft.Count + _tempRight.Count;
            for (int i = 0; i < _length; i++)
            {
                if (_tempLeft.Count > 0 && _tempRight.Count > 0)
                {
                    _queue.Enqueue(comparer.Compare(_tempLeft.Peek(), _tempRight.Peek()) <= 0 ? _tempLeft.Dequeue() : _tempRight.Dequeue());
                }
                else if (_tempLeft.Count > 0)
                {
                    _queue.Enqueue(_tempLeft.Dequeue());
                }
                else if (_tempRight.Count > 0)
                {
                    _queue.Enqueue(_tempRight.Dequeue());
                }
            }
            _listQueue.Enqueue(_queue);
        }


        if (_listQueue.Count > 1)
        {
            _listQueue.InternalMergeSortA(comparer);
        }
        collectionQueue.Clear();
       
        foreach (var item in _listQueue)
        {
            collectionQueue.Enqueue(item);
        }
    }

}

測試用例

[TestMethod]
    public void TestMethod3()
    {
        List<int> list = new List<int>() { 23, 42, 4, 16, 8, 23, 15, 3, 9, 55, 0, 23, 34, 12, 2, 46, 25, 25 };
        Queue<int> _queue = new Queue<int>(list);
        _queue.MergeSortA();
        Assert.Fail();
    }

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

-Advertisement-
Play Games
更多相關文章
  • 在保密你的伺服器和數據,防備當前複雜的攻擊,SQL Server有你需要的一切。但在你能有效使用這些安全功能前,你需要理解你面對的威脅和一些基本的安全概念。這篇文章提供了基礎,因此你可以對SQL Server里的安全功能充分利用,不用在面對特定威脅,不能保護你數據的功能上浪費時間。 從讓人眼花繚亂的 ...
  • 介紹 grep是一個功能強大的文本搜索命令,可以用它來搜索某個文件中是否包含指定的搜索內容,它可以利用正則表達式來做複雜的篩選操作,它還可以為其它命令傳輸給管道的篩選,比如我們常用到的分析單個進程的操作就是會利用它“ps -ef|grep command”。 語法 grep [OPTION]... ...
  • 一、關於CentOS系統介紹 CentOS(Community Enterprise Operating System,中文意思是:社區企業操作系統)是Linux發行版之一,它是來自於Red Hat Enterprise Linux依照開放源代碼規定釋出的源代碼所編譯而成。基於Red Hat持續升級 ...
  • 1 引言 相關源碼下載地址:http://www.jinhusns.com/Products/Download/?type=xcj 1.1 目的 用於社會化開發平臺的架構設計指導,闡述基礎設施及關鍵技術構件、業務構件的設計思想及具體實現。 讀者包括但不限於社會化開發平臺的研發人員,使用社會化開發平臺 ...
  • SSIO的更新 在SSIO上增加了UDP通訊方式,可以到Github上下載源代碼。在原來的項目中,遠端的設備與中心站的數據交互並沒有使用過UDP方式。這種短連接的通訊鏈路,不容易維護,主要體現在:(1)持續的數據交互能力。(2)對現場設備進行長時間的維護和校準。(3)SSIO要協調設備、IO和控制方 ...
  • 集合:可以使用集合來維護對象組。 C#中的數組實現為 System.Array 類的實例,它們只是集合類(Collection Classes)中的一種類型。集合類一般用於處理對象列表,其功能比簡單數組要多,功能大多是通過實現 System.Collections 名稱空間中的介面而獲得的, 因此集 ...
  • 首先先套用一下書中對於委托的描述 什麼是委托,法庭上律師為當事人辯護,他真正執行的是當事人的陳詞,律師就相當於一個委托對象,而當事人則委托律師對象為自己辯護。 C#中的委托概念就好比律師對象,它是一個類(“委托是類類型”),因為只有類對象才有對象的概念。 C#中的委托可以理解為函數的一個包裝,它可以 ...
  • 1、通過JavaScript獲取本機的時間,自動更新 <script> function displayTime() { var date = new Date(); //日期對象 var now = ""; now = date.getFullYear() + "年"; now = now + ( ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...