自頂向下歸併排序和自底向上的歸併排序

来源:http://www.cnblogs.com/nullzx/archive/2016/10/16/5968170.html
-Advertisement-
Play Games

歡迎探討,如有錯誤敬請指正 如需轉載,請註明出處http://www.cnblogs.com/nullzx/ 1. 歸併排序演算法的使用情景 歸併排序演算法和快速排序演算法是java.util.Arrays中使用的排序算。對於一般的基本數據類型,Arrays.sort函數使用雙軸快速排序演算法,而對於對象類... ...


歡迎探討,如有錯誤敬請指正

如需轉載,請註明出處http://www.cnblogs.com/nullzx/


1. 歸併排序演算法的使用情景

歸併排序演算法和快速排序演算法是java.util.Arrays中使用的排序算。對於一般的基本數據類型,Arrays.sort函數使用雙軸快速排序演算法,而對於對象類型使用歸併排序(準確的說使用的是TimSort排序演算法,它是歸併排序的優化版本)。這樣做的原因有兩點,第一個原因,歸併排序是穩定的,而快速排序不是穩定的。第二個原因,對於基本數據類型,排序的穩定性意義不大,但對於複合數據類型(比如對象)排序的穩定性就能幫助我們保持排序結果的某些性質。

2. 自頂向下的歸併排序

自頂向下的排序演算法就是把數組元素不斷的二分,直到子數組的元素個數為一個,因為這個時候子數組必定是已有序的,然後將兩個有序的序列合併成一個新的有序的序列,兩個新的有序序列又可以合併成另一個新的有序序列,以此類推,直到合併成一個有序的數組。

image

為了體現歸併的排序的穩定性,我們的代碼使用java的泛型來實現對任意對象的排序。

	public static <T extends Comparable<? super T>> 
	void MergeSortUpToDown(T[] A){
		@SuppressWarnings("unchecked")
		//創建合併兩個有序序列的輔助數組
		T[] aux = (T[])Array.newInstance(A.getClass().getComponentType(), A.length);
		mergeSortUpToDown0(A, aux, 0, A.length-1);
	}
	
	public static <T extends Comparable<? super T>> 
	void mergeSortUpToDown0(T[] A, T[] aux, int start, int end){
		if(start == end)
			return;
		int mid = (start+end)/2;
		mergeSortUpToDown0(A, aux, start, mid);
		mergeSortUpToDown0(A, aux, mid+1, end);
		//複製到輔助數組中,此時[start,mid] [mid+1, end]兩個子數組已經有序
		System.arraycopy(A, start, aux, start, end - start + 1);
		//然後歸併回來
		int i = start, j = mid+1, k;
		for(k = start; k <= end; k++){
			if(i > mid){
				A[k] = aux[j++];
			}else
			if(j > end){
				A[k] = aux[i++];
			}else
			if(aux[i].compareTo(aux[j]) <= 0){
				A[k] = aux[i++];
			}else{
				A[k] = aux[j++];
			}
		}
	}

3. 自底向上的歸併排序

自底向上的歸併排序演算法的思想就是數組中先一個一個歸併成兩兩有序的序列,兩兩有序的序列歸併成四個四個有序的序列,然後四個四個有序的序列歸併八個八個有序的序列,以此類推,直到,歸併的長度大於整個數組的長度,此時整個數組有序。需要註意的是數組按照歸併長度劃分,最後一個子數組可能不滿足長度要求,這個情況需要特殊處理。自頂下下的歸併排序演算法一般用遞歸來實現,而自底向上可以用迴圈來實現。

image

	//自底向上歸併排序
	public static <T extends Comparable<? super T>> void MergeSortDownToUp(T[] A){
		@SuppressWarnings("unchecked")
		T[] aux = (T[])Array.newInstance(A.getClass().getComponentType(), A.length);
		int len,i,j,k,start,mid,end;
		//len表示歸併子數組的長度,1表示,一個一個的歸併,歸併後的長度為2,2表示兩個兩個的歸併,歸併後的長度為4,以此類推
		for(len = 1; len < A.length; len = 2*len){
			//複製到輔助數組中
			System.arraycopy(A, 0, aux, 0, A.length);
			//按照len的長度歸併回A數組,歸併後長度翻倍
			for(start = 0; start < A.length; start = start+2*len){
				mid = start + len - 1;
				//對於數組長度不滿足2的x次冪的數組,mid可能會大於end
				end = Math.min(start + 2*len - 1, A.length-1);
				i = start; 
				//mid大於end時,j必然大於end,所以不會引起越界訪問
				j = mid+1;
				//[start,mid] [mid+1, end]
				for(k = start; k <= end; k++){
					if(i > mid){
						A[k] = aux[j++];
					}else
					if(j > end){
						A[k] = aux[i++];
					}else
					if(aux[i].compareTo(aux[j]) < 0){
						A[k] = aux[i++];
					}else{
						A[k] = aux[j++];
					}
				}
			}
		}
	}

4. 參考文章

[1] 演算法(第四版)RobertSedgewick


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

-Advertisement-
Play Games
更多相關文章
  • ...
  • **** *************************************************************************** 十月 17, 2016 1:48:19 上午 org.hibernate.annotations.common.reflection.ja ...
  • Java泛型中有存在一種方式叫做類型擦除,也就是說泛型在編譯期間進行類型檢驗上做到有效安全,但是在運行當中,會將該泛型類型用頂層父類(若無繼承關係則用Object)代替,然後再進行強轉換成目標類型,這種類型擦除也存在在泛型方法中,但是方法的擦除帶來了兩個複雜的問題。 在類型擦除之後,代碼演變成如下的 ...
  • ...
  • Python3 數字(Number) 定義:a=1 特性: 1.只能存放一個值 2.一經定義,不可更改 3.直接訪問 分類:整型,長整型,布爾,浮點,複數 python2.*與python3.*關於整型的區別 Python 數字數據類型用於存儲數值。 數據類型是不允許改變的,這就意味著如果改變數字數 ...
  • 隨筆簡介 spring版本:4.3.2.RELEASE+spring security 版本:4.1.2.RELEASE(其它不做說明) 所展示內容全部用註解配置 springmvc已經配置好,不作說明 會涉及到springmvc,spel,el的東西,不熟悉的同學可以先去看一下這方面內容,特別是s ...
  • 1.簡介 通常在R中從來進行分析和展現的數據都是以基本的格式保存的,如.csv或者.Rdata,然後使用.Rmd文件來進行分析的呈現。通過這個方式,分析師不僅可以呈現他們的統計分析的結果,還可以直接生成pdf和html文件,節省了大量的時間。但是,當你想要給其他人參閱你的文檔的時候,你就需要編譯.R ...
  • 、 十月 16, 2016 11:11:12 下午 org.hibernate.annotations.common.reflection.java.JavaReflectionManager <clinit>INFO: HCANN000001: Hibernate Commons Annotati ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...