欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

深入探究TimSort對(duì)歸并排序算法的優(yōu)化及Java實(shí)現(xiàn)

 更新時(shí)間:2016年05月04日 11:27:49   作者:Lizo_Is_Me  
這篇文章主要介紹了TimSort歸并排序的優(yōu)化及Java實(shí)現(xiàn),TimSort 是一個(gè)歸并排序做了大量?jī)?yōu)化的版本,需要的朋友可以參考下

簡(jiǎn)介
MergeSort對(duì)已經(jīng)反向排好序的輸入時(shí)復(fù)雜度為O(n^2),而timsort就是針對(duì)這種情況,對(duì)MergeSort進(jìn)行優(yōu)化而產(chǎn)生的,平均復(fù)雜度為n*O(log n),最好的情況為O(n),最壞情況n*O(log n)。并且TimSort是一種穩(wěn)定性排序。思想是先對(duì)待排序列進(jìn)行分區(qū),然后再對(duì)分區(qū)進(jìn)行合并,看起來(lái)和MergeSort步驟一樣,但是其中有一些針對(duì)反向和大規(guī)模數(shù)據(jù)的優(yōu)化處理。

歸并排序的優(yōu)化思想
歸并排序有以下幾點(diǎn)優(yōu)化方法:

和快速排序一樣,對(duì)于小數(shù)組可以使用插入排序或者選擇排序,避免遞歸調(diào)用。
在merge()調(diào)用之前,可以判斷一下a[mid]是否小于等于a[mid+1]。如果是的話那么就不用歸并了,數(shù)組已經(jīng)是有序的。原因很簡(jiǎn)單,既然兩個(gè)子數(shù)組已經(jīng)有序了,那么a[mid]是第一個(gè)子數(shù)組的最大值,a[mid+1]是第二個(gè)子數(shù)組的最小值。當(dāng)a[mid]<=a[mid+1]時(shí),數(shù)組整體有序。
為了節(jié)省將元素復(fù)制到輔助數(shù)組作用的時(shí)間,可以在遞歸調(diào)用的每個(gè)層次交換原始數(shù)組與輔助數(shù)組的角色。
在merge()方法中的歸并過(guò)程需要判斷i和j是否已經(jīng)越界,即某半邊已經(jīng)用盡。可以用另一種方式,去掉檢測(cè)是否某半邊已經(jīng)用盡的代碼。具體步驟是將數(shù)組a[]的后半部分以降序的方式復(fù)制到aux[],然后從兩端歸并。對(duì)于數(shù)組{1,2,3}和{2,3,5},第一個(gè)子數(shù)組照常復(fù)制,第二個(gè)則從后往前復(fù)制,最終aux[]中的元素為{1,2,3,5,3,2}。這種方法的缺點(diǎn)是使得歸并排序變?yōu)椴环€(wěn)定排序。代碼實(shí)現(xiàn)如下:

void merge(int[] a, int lo, int mid, int hi, int[] aux) {
for (int k = lo; k <= mid; k++) {
  aux[k] = a[k];
}
for (int k = mid + 1;k <= hi; k++) {
  aux[k] = a[hi - k + mid + 1];
}
int i = lo, j = hi;   //從兩端往中間
for (int k = lo; k <= hi; k++)
  if (aux[i] <= aux[j]) a[k] = aux[i++];
  else a[k] = aux[j--];
}

TimSort的步驟

分區(qū)

分區(qū)的思想是掃描一次數(shù)組,把連續(xù)正序列(如果是升序排序,那么正序列就是升序序列),或者【嚴(yán)格】(保證排序算法的穩(wěn)定性)的反序列做為一個(gè)分區(qū)(run),如果是反序列,把分區(qū)里的元素反轉(zhuǎn)一下。 例如
1,2,3,6,4,5,8,6,4 劃分分區(qū)結(jié)果為
[1,2,3,6],[4,5,8],[6,4]
然后反轉(zhuǎn)反序列
[1,2,3,6],[4,5,8],[4,6]

合并

考慮一個(gè)極端的例子,比如分區(qū)的長(zhǎng)度分別為 10000,10,1000,10,10,我們當(dāng)然希望是先讓10個(gè)10合并成20, 20和1000合并成1020如此下去, 如果從從左往右順序合并的話,每次都用到10000這個(gè)數(shù)組和去小的數(shù)組合并,代價(jià)太大了。所以我們可以用一個(gè)策略來(lái)優(yōu)化合并的順序。

實(shí)例

以java中的ComparableTimSort.sort()為例子, 用了一個(gè)run stack來(lái)確定是否應(yīng)該合并,

    if (nRemaining < MIN_MERGE) {
      int initRunLen = countRunAndMakeAscending(a, lo, hi);
      binarySort(a, lo, hi, lo + initRunLen);
      return;
    }


小于MIN_MERGE(32)的排序,分區(qū)后直接用二分插入排序

int minRun = minRunLength(nRemaining);
    do {
      //找出下一個(gè)分區(qū)的起始位置,同時(shí)也對(duì)反向序列做了翻轉(zhuǎn)處理
      int runLen = countRunAndMakeAscending(a, lo, hi);

      //保證run stack中的run的都大于minRun ,如果當(dāng)前分區(qū)太小,就從后面取出元素補(bǔ)足
      if (runLen < minRun) {
        int force = nRemaining <= minRun ? nRemaining : minRun;
        binarySort(a, lo, lo + force, lo + runLen);
        runLen = force;
      }

      //把run放入 run stack中
      ts.pushRun(lo, runLen);
      //判斷是否應(yīng)該合并,i是從棧頂開(kāi)始的,知道不能合并為止
      //1. runLen[i - 3] > runLen[i - 2] + runLen[i - 1] 
      //2. runLen[i - 2] > runLen[i - 1]
      ts.mergeCollapse();


      lo += runLen;
      nRemaining -= runLen;
    } while (nRemaining != 0);

    // Merge all remaining runs to complete sort
    assert lo == hi;
    //合并剩下的run
    ts.mergeForceCollapse();
    assert ts.stackSize == 1;


在看里面的一個(gè)比較重要的函數(shù)

/**
* 如果后2個(gè)run的長(zhǎng)度加起來(lái)比前面一個(gè)長(zhǎng),則使用中間位置的run和前后長(zhǎng)度更短的run一個(gè)合并
* 如果后2個(gè)run的長(zhǎng)度加起來(lái)比前面一個(gè)短,則把后面2個(gè)run合并
*/
 private void mergeCollapse() {
    while (stackSize > 1) {
      int n = stackSize - 2;
      if (n > 0 && runLen[n-1] <= runLen[n] + runLen[n+1]) {
        if (runLen[n - 1] < runLen[n + 1])
          n--;
        mergeAt(n);
      } else if (runLen[n] <= runLen[n + 1]) {
        mergeAt(n);
      } else {
        break; // Invariant is established
      }
    }
  }

相關(guān)文章

  • java dom4j解析xml用到的幾個(gè)方法

    java dom4j解析xml用到的幾個(gè)方法

    這篇文章主要介紹了java dom4j解析xml用到的幾個(gè)方法,有需要的朋友可以參考一下
    2013-12-12
  • 淺談@FeignClient中name和value屬性的區(qū)別

    淺談@FeignClient中name和value屬性的區(qū)別

    這篇文章主要介紹了@FeignClient中name和value屬性的區(qū)別,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-07-07
  • Spring之InitializingBean接口和DisposableBean接口的使用

    Spring之InitializingBean接口和DisposableBean接口的使用

    這篇文章主要介紹了Spring之InitializingBean接口和DisposableBean接口的使用方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-01-01
  • 基于springioc bean 的幾個(gè)屬性介紹

    基于springioc bean 的幾個(gè)屬性介紹

    下面小編就為大家?guī)?lái)一篇基于springioc bean 的幾個(gè)屬性介紹。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就想給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2017-09-09
  • java rocketmq--消息的產(chǎn)生(普通消息)

    java rocketmq--消息的產(chǎn)生(普通消息)

    這篇文章主要介紹了java rocketmq--消息的產(chǎn)生(普通消息),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,,需要的朋友可以參考下
    2019-06-06
  • Java try catch finally異常處理組合詳解

    Java try catch finally異常處理組合詳解

    這篇文章主要介紹了Java try catch finally異常處理組合詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-05-05
  • Java實(shí)現(xiàn)帶圖形界面的聊天程序

    Java實(shí)現(xiàn)帶圖形界面的聊天程序

    這篇文章主要為大家詳細(xì)介紹了Java實(shí)現(xiàn)帶圖形界面的聊天程序,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-06-06
  • Spring Boot如何優(yōu)化內(nèi)嵌的Tomcat示例詳解

    Spring Boot如何優(yōu)化內(nèi)嵌的Tomcat示例詳解

    spring boot默認(rèn)web程序啟用tomcat內(nèi)嵌容器,監(jiān)聽(tīng)8080端口,下面這篇文章主要給大家介紹了關(guān)于Spring Boot如何優(yōu)化內(nèi)嵌Tomcat的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),需要的朋友可以參考借鑒,下面來(lái)一起看看吧。
    2017-09-09
  • Java異常之圖書(shū)管理系統(tǒng)

    Java異常之圖書(shū)管理系統(tǒng)

    這篇文章主要為大家詳細(xì)介紹了Java異常之圖書(shū)管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-11-11
  • java用接口、多態(tài)、繼承、類計(jì)算三角形和矩形周長(zhǎng)及面積的方法

    java用接口、多態(tài)、繼承、類計(jì)算三角形和矩形周長(zhǎng)及面積的方法

    這篇文章主要介紹了java用接口、多態(tài)、繼承、類計(jì)算三角形和矩形周長(zhǎng)及面積的方法,涉及java面向?qū)ο笾蓄?、接口、多態(tài)等的使用技巧,需要的朋友可以參考下
    2015-05-05

最新評(píng)論