Java排序算法 - 歸并排序

歸并排序

思路:使用分治思想,將數組一直拆分,直到拆分成一個元素,此時每一個元素都相當于一個有序的數組,之后再將每兩個數組合并成一個有序數組,一直到所有的數組都合并完成,最后只剩一個數組時,就完成了排序。

步驟如下

首先,我們先來學習一下如何將兩個已經排序的數組按照成一個數組。
例如,這里有兩個排好序的序列arr1和arr2。


數組合并

如何將兩個有序的數組合并成一個的數組呢?
第一,先創建一個等于兩個數組長度之和的數組arr,n、m、k分別指向arr1、arr2,arr的第0個元素。


數組合并

之后進行比較arr1[n]和arr2[m]的大小,把較小的移入arr中。例如8<5,將5移入arr[k]位置,之后m++,k++。


數組合并

當n或者m到達數組結尾時,則將另一個數組剩余的元素,直接都放到arr中,例如此時將16和19直接放入arr中,不用進行比較了。
m到達數組尾部

合并完成

至此完成數組的合并。

合并代碼如下

private void mergeArray(T[] array, int begin, int mid, int end){
        if(array[mid].compareTo(array[mid + 1]) < 0){
            return;
        }else{
            T[] temp = (T[])new Comparable[end - begin + 1];
            int n = begin;
            int m = mid + 1;
            int k = 0;
            while((n <= mid) && (m <= end)){
                if(array[n].compareTo(array[m]) < 0){
                    temp[k++] = array[n++];
                }else{
                    temp[k++] = array[m++];
                }
            }

            while(n <= mid){
                temp[k++] = array[n++];
            }

            while(m <= end){
                temp[k++] = array[m++];
            }

            for(int i = 0; i < temp.length; i++){
                array[begin + i] = temp[i];
            }
        }
    }

好,學習完了合并,進入正題,將數組拆分。
先獲得數組的中間下標(int mid = (begin + end)/ 2),數組拆分是一個遞歸的過程,將begin作為第一個數組的起始,mid作為結尾,mid+1作為另一個數組的起始,end作為結尾,重新傳入歸并排序方法進行繼續拆分。遞歸拆分過程如下

拆分結束

之后將每一個元素看成排好序的數組,使用上面的方法兩兩合并。

歸并方法代碼

private void mergeSort(T[] array, int begin, int end){
        if(begin < end){
            int mid = (begin + end) / 2;
            mergeSort(array, begin, mid);
            mergeSort(array, mid+1, end);
            mergeArray(array, begin, mid, end);
        }
    }

完整代碼如下

/**
 * Created by ShouJingGuo on 2018/3/18.
 */
public class MergeSort<T extends Comparable<T>>{
    public void mergeSort(T[] array){
        mergeSort(array, 0, array.length-1);
    }

    private void mergeSort(T[] array, int begin, int end){
        if(begin < end){
            int mid = (begin + end) / 2;
            mergeSort(array, begin, mid);
            mergeSort(array, mid+1, end);
            mergeArray(array, begin, mid, end);
        }
    }

    private void mergeArray(T[] array, int begin, int mid, int end){
        if(array[mid].compareTo(array[mid + 1]) < 0){
            return;
        }else{
            T[] temp = (T[])new Comparable[end - begin + 1];
            int n = begin;
            int m = mid + 1;
            int k = 0;
            while((n <= mid) && (m <= end)){
                if(array[n].compareTo(array[m]) < 0){
                    temp[k++] = array[n++];
                }else{
                    temp[k++] = array[m++];
                }
            }

            while(n <= mid){
                temp[k++] = array[n++];
            }

            while(m <= end){
                temp[k++] = array[m++];
            }

            for(int i = 0; i < temp.length; i++){
                array[begin + i] = temp[i];
            }
        }
    }

    public static void main(String[] args) {
        Integer arr[] = {10,50,24,11,68,20,41,0,24,25,4,7,94,15,5,44,66};
        MergeSort<Integer> mergeSort= new MergeSort<>();
        mergeSort.mergeSort(arr);
        System.out.println(Arrays.toString(arr));
    }
}
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,443評論 6 532
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,530評論 3 416
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,407評論 0 375
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,981評論 1 312
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,759評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,204評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,263評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,415評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,955評論 1 336
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,782評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,983評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,528評論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,222評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,650評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,892評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,675評論 3 392
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,967評論 2 374

推薦閱讀更多精彩內容