首頁 後端開發 php教程 排序演算法—歸併排序【附代碼】

排序演算法—歸併排序【附代碼】

Aug 22, 2019 pm 05:06 PM
排序演算法

排序演算法—歸併排序【附代碼】

什麼是歸併排序?

  歸併排序簡單來講,就是將兩個有序的序列整合到一起。

推薦教學:PHP影片教學

#如何將兩個有順序的序列整合在一起呢?

  那麼我們假設,現在有 M={m1 ,m2,m3,....,mx}序列和 N = {n1,n2,n3,...., ny}序列,這兩個序列已經是有序的序列,首先創建一個空序列 K = {},那麼接著將 m1 和 n1 進行比較,加入 m1 < n1 那麼將 m1  K 序列中,然後 M 序列中,然後 M1 序列遊標後移,即下次將進行 m2 和 n1 的比較,直到全部比較完畢,並填入序列 K 中。

既然歸併排序是整合有序序列,那麼豈不是不能排序無序序列,這條件也太苛刻了點吧?

  歸併排序是建立在分治法的基礎上進行操作的,也就是可以將一個完全無序的序列進行無線分割以達到有序序列,例如:現在有M={m1 ,m2,m3,....,mx},M序列是完全無序的序列,那麼此時可以將 M 序列分成若干個小序列-{m1,m2},{m3,m4 }....{mx-1,mx};此時這些序列就是有序的,那麼就可以透過歸併操作進行排序,所以歸併排序也可以排序無序序列,且速度僅次於快速排序,屬於穩定排序。

如何分割序列?

  上個問題已經提過,歸併排序是建立在分治的基礎上進行操作的,其中分治的體現就體現在分割序列上,比如:現在有M ={m1 ,m2,m3,....,mx},第一次分割:M1 = {m1,m2,...,m(x 1)/2},M2 = {m(x 1)/ 2 1,m(x 1)/2 2,...,mx},第一次分割之後得到 M1 和 M2 序列,然後再分割 M1 和 M2 ,分割若干次之後得到 x/2 x%2 個序列,然後再逐一進行歸併操作。

此演算法特定步驟:

  1、規定首指標 left 與mid(left指向序列1的首元素,right 指向序列2的首元素)

  2、比較 left 和 mid 的大小,將小的元素放入新建的空間中

  3、重複步驟2,直到其中一個序列遍歷完畢

  4、將剩餘的未加入新空間中的元素直接複製貼上進新空間

該演算法的核心步驟:

  1、使用分治法(遞歸)分割序列

  2、比較 left 和 mid 的大小, 將小的元素的添加進入新建空間

敘述完畢,貼上程式碼:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace 排序__归并排序
{
    class 归并
    {

        public static int[] arr = { 6, 202, 301, 100, 38, 8, 1 ,-1,1000};
        static void Main(string[] args)
        {
            Sort(arr, 0, arr.Length -1);
            foreach (var item in arr)
            {
                Console.Write(item + "  ");
            }
            Console.ReadKey();
        }

        public static void Sort(int []a,int left,int right)
        {
            if (left >= right) return;
            else
            {
                int mid = (left + right) / 2;                //@1
                Sort(a, left, mid);
                Sort(a, mid + 1, right);
                MergeSort(a, left, mid, right);
            }
        }

        public static void MergeSort(int []a,int left,int mid,int right)
        {
            int[] Arr = new int[right - left + 1];
            int l = left, m = mid +1 , k = 0;           //@2
            while ( m <= right && l <= mid )          //@3
            {
                if (a[l] > a[m]) Arr[k++] = a[m++];
                else Arr[k++] = a[l++];
            }
            while (m < right +1)                           //@4
            {
                Arr[k++] = a[m++];
            }
            while (l < mid +1 )  Arr[k++] = a[l++];        //@4
            for (k = 0, l = left; l < right + 1; k++, l++) a[l] = Arr[k];
        }
    }
}
登入後複製

程式碼解讀:

##  @1 :Sort()函數完成了序列的分割,每一次遞迴都會將序列分成兩半,直到無法分隔為止。

  @2 :l 代表序列1的首元素,m 代表序列2的首元素,k代表新建數組Arr的已有元素個數

  @3 :序列1的首元素和序列2的首元素進行比較,將小的放入 Arr 中,且序列遊標後移,直到其中一個序列的元素遍比較完畢

  @4 :在序列1 和序列2的比較過程中,可能出現序列1已經全部加入了 Arr 中,但是序列2還沒有,則將剩餘的未比較的元素直接複製進入 Arr 中

##總結:

  以上程式碼並不是真正意義上的分割數組,只是做了一個邏輯分割,沒有像其他程式碼一樣將分割後的數組填入一個新的數組,那樣做的話會對速度和記憶體產生一點影響,雖然微乎其微,但總歸是沒這麼好的,在歸併操作上,需要注意的是參數不要越界(我當時就想了很久為什麼 @2 上面的m 要等於 mid 1 而不是 mid ,其實道理很簡單,因為mid是屬於左序列,從 mid 1 開始才屬於右序列,將m = mid 1  改成 m = mid 不是不行,只是後面的程式碼也需要改,但是沒有必要多做一次無用比較,這個問題我當時真是想了半天...),其實只要理清楚其中的邏輯關係,這樣程式碼就能做到信手拈來。

以上是排序演算法—歸併排序【附代碼】的詳細內容。更多資訊請關注PHP中文網其他相關文章!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

<🎜>:泡泡膠模擬器無窮大 - 如何獲取和使用皇家鑰匙
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
北端:融合系統,解釋
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
Mandragora:巫婆樹的耳語 - 如何解鎖抓鉤
3 週前 By 尊渡假赌尊渡假赌尊渡假赌

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

熱門話題

Java教學
1666
14
CakePHP 教程
1425
52
Laravel 教程
1323
25
PHP教程
1272
29
C# 教程
1251
24
快手雙邊市場的複雜實驗設計問題 快手雙邊市場的複雜實驗設計問題 Apr 15, 2023 pm 07:40 PM

一、問題背景1、雙邊市場實驗介紹雙邊市場,即平台,包含生產者與消費者兩方參與者,雙方相互促進。例如快手有影片的生產者,影片的消費者,兩種身分可能有一定程度重疊。雙邊實驗是在生產者和消費者端組合分組的實驗方式。雙邊實驗有以下優點:(1)可以同時偵測新策略對兩方面的影響,例如產品DAU和上傳作品人數變化。雙邊平台往往有跨邊網路效應,讀者越多,作者越活躍,作者越活躍,讀者也會跟著增加。 (2)可以檢測效果溢出和轉移。 (3)幫助我們更好得理解作用的機制,AB實驗本身不能告訴我們原因和結果之間的關係,只

Vue技術開發中如何進行資料篩選與排序 Vue技術開發中如何進行資料篩選與排序 Oct 09, 2023 pm 01:25 PM

Vue技術開發中如何進行資料篩選和排序在Vue技術開發中,資料篩選和排序是非常常見且重要的功能。透過資料篩選和排序,我們可以快速查詢和展示我們需要的信息,提高用戶體驗。本文將介紹在Vue中如何進行資料篩選和排序,並提供具體的程式碼範例,幫助讀者更好地理解和運用這些功能。一、資料篩選資料篩選是指依照特定的條件篩選出符合要求的資料。在Vue中,我們可以透過comp

如何使用C++中的基數排序演算法 如何使用C++中的基數排序演算法 Sep 19, 2023 pm 12:15 PM

如何使用C++中的基數排序演算法基數排序演算法是一種非比較性的排序演算法,它透過將待排序的元素分割成一組有限的數字位元來完成排序。在C++中,我們可以使用基數排序演算法來對一組整數進行排序。以下我們將詳細討論如何實作基數排序演算法,並附上具體的程式碼範例。演算法思想基數排序演算法的想法是將待排序的元素分割成一組有限的數字位,然後依序對每個位上的元素進行排序。在每個位元上的排序完

谷歌借AI打破十年排序演算法封印,每天被執行數萬億次,網友卻說是最不切實際的研究? 谷歌借AI打破十年排序演算法封印,每天被執行數萬億次,網友卻說是最不切實際的研究? Jun 22, 2023 pm 09:18 PM

整理|核子可樂,褚杏娟接觸過基礎電腦科學課程的朋友們,肯定都曾親自動手設計排序演算法——也就是藉助代碼將無序列表中的各個條目按升序或降序方式重新排列。這是個有趣的挑戰,可行的操作方法也多元。人們曾投入大量時間探索如何更有效率地完成排序任務。作為一項基礎操作,大多數程式語言的標準庫中都內建有排序演算法。世界各地的程式碼庫中使用了許多不同的排序技術和演算法來在線組織大量數據,但至少就與LLVM編譯器配套使用的C++庫而言,排序程式碼已經有十多年沒有任何變化了。近日,GoogleDeepMindAI小組如今開發出一

如何實作C#中的選擇排序演算法 如何實作C#中的選擇排序演算法 Sep 20, 2023 pm 01:33 PM

如何實現C#中的選擇排序演算法選擇排序(SelectionSort)是一種簡單直觀的排序演算法,其基本思想是每次從待排序元素中選擇最小(或最大)的元素,放到已排序的序列末尾。透過重複這個過程,直到所有元素都排序完成。下面我們來詳細了解如何在C#中實作選擇排序演算法,同時附上具體的程式碼範例。建立選擇排序方法首先,我們需要建立一個用於實作選擇排序的方法。此方法接受一

Swoole進階:如何使用多執行緒實作高速排序演算法 Swoole進階:如何使用多執行緒實作高速排序演算法 Jun 14, 2023 pm 09:16 PM

Swoole是一款基於PHP語言的高效能網路通訊框架,它支援多種非同步IO模式和多種高階網路協定的實作。在Swoole的基礎上,我們可以利用其多執行緒功能來實現高效率的演算法運算,例如高速排序演算法。高速排序演算法(QuickSort)是一種常見的排序演算法,透過定位一個基準元素,將元素分成兩個子序列,小於基準元素的放在左側,大於等於基準元素的放在右側,再對左右子序列遞迴

不同 PHP 數組排序演算法的應用場景探討 不同 PHP 數組排序演算法的應用場景探討 Apr 28, 2024 am 09:39 AM

針對不同場景,選擇合適的PHP數組排序演算法至關重要。冒泡排序適用於小規模數組無穩定性要求的情況;快速排序在大多數情況下時間複雜度最低;歸併排序穩定性高,適用於需要穩定結果的場景;選擇排序適用於無穩定性要求的情況;堆排序高效率找出最大或最小值。透過實戰案例比較,快速排序在時間效率上優於其他演算法,但需要考慮穩定性時應選擇歸併排序。

數組的排序演算法有哪些? 數組的排序演算法有哪些? Jun 02, 2024 pm 10:33 PM

數組排序演算法用於按特定順序排列元素。常見的演算法類型包括:冒泡排序:透過比較相鄰元素交換位置。選擇排序:找出最小元素交換到目前位置。插入排序:逐一插入元素到正確位置。快速排序:分治法,選擇樞紐元素劃分數組。合併排序:分治法,遞歸排序和合併子數組。

See all articles