亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

細(xì)致解讀希爾排序算法與相關(guān)的Java代碼實(shí)現(xiàn)

 更新時(shí)間:2016年05月04日 17:11:00   作者:飛翔的貓咪  
這篇文章主要介紹了希爾排序算法與相關(guān)的Java代碼實(shí)現(xiàn),希爾排序的時(shí)間復(fù)雜度根據(jù)步長(zhǎng)序列的不同而不同,需要的朋友可以參考下

希爾排序(Shell's sort)是一種非?!吧衿妗钡呐判蛩惴?。說(shuō)它“神奇”,是因?yàn)闆](méi)有任何人能清楚地說(shuō)明它的性能到底能到什么情況。希爾排序因DL.Shell于1959年提出而得名。自從C. A. R. Hoare在1962年提出快速排序后,由于其更為簡(jiǎn)單,一般采用快速排序。但是,不少數(shù)學(xué)家們還是孜孜不倦地尋找希爾排序的最佳復(fù)雜度。作為普通程序員,我們可以學(xué)習(xí)下希爾的思路。
順便說(shuō)一句,在希爾排序出現(xiàn)之前,計(jì)算機(jī)界普遍存在“排序算法不可能突破O(n2)”的觀點(diǎn)。希爾排序的出現(xiàn)打破了這個(gè)魔咒,很快,快速排序等算法相繼問(wèn)世。從這個(gè)意義上說(shuō),希爾排序帶領(lǐng)我們走向了一個(gè)新的時(shí)代。

算法概述/思路
希爾排序的提出,主要基于以下兩點(diǎn):
1.插入排序算法在數(shù)組基本有序的情況下,可以近似達(dá)到O(n)復(fù)雜度,效率極高。
2.但插入排序每次只能將數(shù)據(jù)移動(dòng)一位,在數(shù)組較大且基本無(wú)序的情況下性能會(huì)迅速惡化。

基于此,我們可以使用一種分組的插入排序方法,具體做法是:(以一個(gè)16元素大小的數(shù)組為例)
1.選擇一個(gè)增量delta,該增量大于1,從數(shù)組中按此增量選擇出子數(shù)組進(jìn)行一次直接插入排序。例如,若選擇增量為5,則對(duì)下標(biāo)為0,5,10,15的元素進(jìn)行排序。
2.保留該增量delta并依次移動(dòng)首個(gè)元素進(jìn)行直接插入排序,直到一輪完成。對(duì)于上面的例子,則依次對(duì)數(shù)組[1,6,11],[2,7,12],[3,8,13],[4,9,14]進(jìn)行排序。
3.減小增量,不斷重復(fù)上述過(guò)程,直到增量減小為1.顯然,最后一次為直接插入排序。
4.排序完成。
從上面可以看出,增量是不斷減小的,因此,希爾排序又被成為“縮小增量排序”。

代碼實(shí)現(xiàn)

package sort; 
 
public class ShellSortTest { 
  public static int count = 0; 
 
  public static void main(String[] args) { 
 
    int[] data = new int[] { 5, 3, 6, 2, 1, 9, 4, 8, 7 }; 
    print(data); 
    shellSort(data); 
    print(data); 
 
  } 
 
  public static void shellSort(int[] data) { 
    // 計(jì)算出最大的h值 
    int h = 1; 
    while (h <= data.length / 3) { 
      h = h * 3 + 1; 
    } 
    while (h > 0) { 
      for (int i = h; i < data.length; i += h) { 
        if (data[i] < data[i - h]) { 
          int tmp = data[i]; 
          int j = i - h; 
          while (j >= 0 && data[j] > tmp) { 
            data[j + h] = data[j]; 
            j -= h; 
          } 
          data[j + h] = tmp; 
          print(data); 
        } 
      } 
      // 計(jì)算出下一個(gè)h值 
      h = (h - 1) / 3; 
    } 
  } 
 
  public static void print(int[] data) { 
    for (int i = 0; i < data.length; i++) { 
      System.out.print(data[i] + "\t"); 
    } 
    System.out.println(); 
  } 
 
} 

運(yùn)行結(jié)果:

5  3  6  2  1  9  4  8  7   
1  3  6  2  5  9  4  8  7   
1  2  3  6  5  9  4  8  7   
1  2  3  5  6  9  4  8  7   
1  2  3  4  5  6  9  8  7   
1  2  3  4  5  6  8  9  7   
1  2  3  4  5  6  7  8  9   
1  2  3  4  5  6  7  8  9 

算法性能/復(fù)雜度
希爾排序的增量數(shù)列可以任取,需要的唯一條件是最后一個(gè)一定為1(因?yàn)橐WC按1有序)。但是,不同的數(shù)列選取會(huì)對(duì)算法的性能造成極大的影響。上面的代碼演示了兩種增量。
切記:增量序列中每?jī)蓚€(gè)元素最好不要出現(xiàn)1以外的公因子?。ê茱@然,按4有序的數(shù)列再去按2排序意義并不大)。
下面是一些常見的增量序列。
第一種增量是最初Donald Shell提出的增量,即折半降低直到1。據(jù)研究,使用希爾增量,其時(shí)間復(fù)雜度還是O(n2)。
第二種增量Hibbard:{1, 3, ..., 2^k-1}。該增量序列的時(shí)間復(fù)雜度大約是O(n^1.5)。
第三種增量Sedgewick增量:(1, 5, 19, 41, 109,...),其生成序列或者是9*4^i - 9*2^i + 1或者是4^i - 3*2^i + 1。

算法穩(wěn)定性
我們都知道插入排序是穩(wěn)定算法。但是,Shell排序是一個(gè)多次插入的過(guò)程。在一次插入中我們能確保不移動(dòng)相同元素的順序,但在多次的插入中,相同元素完全有可能在不同的插入輪次被移動(dòng),最后穩(wěn)定性被破壞,因此,Shell排序不是一個(gè)穩(wěn)定的算法。

算法適用場(chǎng)景
Shell排序雖然快,但是畢竟是插入排序,其數(shù)量級(jí)并沒(méi)有后起之秀--快速排序O(n㏒n)快。在大量數(shù)據(jù)面前,Shell排序不是一個(gè)好的算法。但是,中小型規(guī)模的數(shù)據(jù)完全可以使用它。

相關(guān)文章

  • java實(shí)現(xiàn)的2048游戲完整實(shí)例

    java實(shí)現(xiàn)的2048游戲完整實(shí)例

    這篇文章主要介紹了java實(shí)現(xiàn)的2048游戲,結(jié)合完整實(shí)例形式分析了java實(shí)現(xiàn)2048游戲功能的相關(guān)數(shù)值運(yùn)算、swing組件布局、事件響應(yīng)等相關(guān)操作技巧,需要的朋友可以參考下
    2018-01-01
  • Java讀取圖片EXIF信息的方法

    Java讀取圖片EXIF信息的方法

    這篇文章主要介紹了Java讀取圖片EXIF信息的方法,較為詳細(xì)的分析了圖片EXIF信息的概念、功能及java讀取EXIF信息的實(shí)現(xiàn)技巧,需要的朋友可以參考下
    2015-07-07
  • 基于Java實(shí)現(xiàn)二維碼的生成和解析

    基于Java實(shí)現(xiàn)二維碼的生成和解析

    二維碼其實(shí)就是一種編碼技術(shù),只是這種編碼技術(shù)是用在圖片上了,將給定的一些文字,數(shù)字轉(zhuǎn)換為一張經(jīng)過(guò)特定編碼的圖片。本文將利用Java實(shí)現(xiàn)二維碼的生成和解析,需要的可以參考一下
    2022-08-08
  • 教你如何使用Java輸出各種形狀

    教你如何使用Java輸出各種形狀

    本文小編將向大家介紹的是如何利用Java輸出各種不同的形狀,本文一共介紹了七種有趣的形狀,感興趣的小伙伴趕快收藏起來(lái)吧
    2021-09-09
  • idea進(jìn)程結(jié)束但是項(xiàng)目頁(yè)面正常運(yùn)行怎么辦

    idea進(jìn)程結(jié)束但是項(xiàng)目頁(yè)面正常運(yùn)行怎么辦

    這篇文章主要介紹了idea進(jìn)程結(jié)束但是項(xiàng)目頁(yè)面正常運(yùn)行怎么辦,很多朋友遇到這樣的情況不知道該如何解決了,下面小編給大家?guī)?lái)了idea進(jìn)程結(jié)束但是項(xiàng)目頁(yè)面正常運(yùn)行的解決方法,需要的朋友可以參考下
    2023-03-03
  • springboot集成druid,多數(shù)據(jù)源可視化,p6spy問(wèn)題

    springboot集成druid,多數(shù)據(jù)源可視化,p6spy問(wèn)題

    這篇文章主要介紹了springboot集成druid,多數(shù)據(jù)源可視化,p6spy問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-01-01
  • SpringBoot實(shí)現(xiàn)啟動(dòng)項(xiàng)目后立即執(zhí)行的方法總結(jié)

    SpringBoot實(shí)現(xiàn)啟動(dòng)項(xiàng)目后立即執(zhí)行的方法總結(jié)

    在項(xiàng)目開發(fā)中某些場(chǎng)景必須要用到啟動(dòng)項(xiàng)目后立即執(zhí)行方式的功能,所以這篇文章就來(lái)和大家聊聊實(shí)現(xiàn)立即執(zhí)行的幾種方法,希望對(duì)大家有所幫助
    2023-05-05
  • Java多線程之線程同步

    Java多線程之線程同步

    這篇文章主要介紹了Java多線程之線程同步,文中有非常詳細(xì)的代碼示例,對(duì)正在學(xué)習(xí)java的小伙伴們有非常好的幫助,需要的朋友可以參考下
    2021-05-05
  • Java每隔兩個(gè)數(shù)刪掉一個(gè)數(shù)問(wèn)題詳解

    Java每隔兩個(gè)數(shù)刪掉一個(gè)數(shù)問(wèn)題詳解

    這篇文章主要介紹了Java每隔兩個(gè)數(shù)刪掉一個(gè)數(shù)問(wèn)題詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2021-04-04
  • 詳解JVM中的GC調(diào)優(yōu)

    詳解JVM中的GC調(diào)優(yōu)

    我們經(jīng)常會(huì)聽到甚至需要自己動(dòng)手去做GC調(diào)優(yōu)。那么GC調(diào)優(yōu)的目的到底是什么呢?讓程序跑得更快?讓GC消耗更少的資源?還是讓程序更加穩(wěn)定?帶著這些疑問(wèn)來(lái)讀一下這篇文章,將會(huì)得到一個(gè)系統(tǒng)的甚至是不一樣的結(jié)果。
    2021-06-06

最新評(píng)論