希爾排序

最后有改進(jìn)版

前情提要
  • 插入排序只在數(shù)據(jù)量小渊胸,數(shù)據(jù)有序程度較高的情況下效率高
  • 基于上述這點(diǎn)柑晒,D.L.Shell于1959年提出了 "縮小增量排序"(Diminishing Increment Sort)滞项,是一種插入排序的優(yōu)化排序算法
核心思想
  • 不同于插入算法眷茁,希爾算法引入了"增量"-gap筷凤,也可以理解為間隔耸携,用處是棵癣,對數(shù)組進(jìn)行按增量進(jìn)行劃分,對每個(gè)劃分進(jìn)行插入排序夺衍,這是為了滿足插入排序高效的一個(gè)條件:數(shù)據(jù)量小

比如狈谊,int[] nums = {4, 3, 5, 7, 6, 8, 9, 1, 2};
nums長度為9,最開始設(shè)置gap = nums.length / 2也就是gap = 4沟沙,此時(shí)數(shù)組劃分為{4, 6, 2}, {3, 8}, {5, 9}, {7, 1}河劝,第一組也就是nums[0] = 4,nums[0 + gap] = 6, nums[0 + 2 * gap] = 2矛紫,第二組就是nums[1] = 3, nums[1 + gap] = 8赎瞎,后面以此類推,注意索引不要越界
然后對每個(gè)組進(jìn)行插入排序颊咬,四個(gè)組排完分別是{2, 4, 6}, {3, 8}, {5, 9}, {1, 7}务甥,此時(shí)數(shù)組nums = {2, 3, 5, 1, 4, 8, 9, 7, 6}可以明顯看出比原始數(shù)組更符合插入排序高效的第二個(gè)條件:數(shù)據(jù)有序程度較高

  • 注意:分組只是邏輯上的分組,在實(shí)際操作中只是特殊處理索引喳篇,比如遍歷一個(gè)分組敞临,for (int i = 0; i < nums.length; i += gap)類似于這樣,把gap當(dāng)成"步長"去遍歷數(shù)組麸澜,就能"分出"每個(gè)組
  • 第一次分組插入排序之后挺尿,把步長gap /= 2砍掉一半,這樣一下,下一輪每個(gè)分組的元素?cái)?shù)量就翻倍(考慮到奇數(shù)個(gè)编矾,所以只能是近似翻倍)

接著例子熟史,第二輪分組,{2, 5, 4, 9, 6}, {3, 1, 8, 7}窄俏,分組插入排序后{2, 4, 5, 6, 9}, {1, 3, 7, 8}以故,數(shù)組{2, 1, 4, 3, 5, 7, 6, 8, 9},而第三輪時(shí)步長gap = 1就變成普通插入排序裆操,但這是數(shù)據(jù)有序性較高怒详,所以性能提高不少

算法實(shí)現(xiàn)

完整代碼
  • 建議先復(fù)制完整代碼再對照著看具體步驟
public class ShellSortDemo {
    private static void swap(int[] nums, int i, int j) {
        int tmp = nums[i];
        nums[i] = nums[j];
        nums[j] = tmp;
    }

    private static void shellSort(int[] nums) {
        for (int gap = nums.length / 2; gap > 0; gap /= 2) {
            for (int i = gap; i < nums.length; i++) {
                for (int j = i; j >= gap; j -= gap) {
                    if (nums[j] < nums[j - gap]) {
                        swap(nums, j, j - gap);
                    } else {
                        break;
                    }
                }
            }
        }
    }
    public static void main(String[] args) {
//        int[] nums = {4, 5, 2, 3, 6, 7, 9, 8, 1, 4, 5, 1, 7, 9, 6, 5, 8};
        int[] nums = new int[2000000];
        for (int i = 0; i < nums.length; i++) {
            nums[i] = (int) (Math.random() * 10000);
        }
//        System.out.println(Arrays.toString(nums));
        long startTime = System.currentTimeMillis();
        shellSort(nums);
        long endTime = System.currentTimeMillis();
        System.out.println("耗時(shí): " + (endTime - startTime) + "ms" );
//        System.out.println(Arrays.toString(nums));
    }
}
具體步驟

整個(gè)過程其實(shí)在草稿紙上用一個(gè)簡單數(shù)組寫一下過程就很清晰了

  • void swap(int[] nums, int i, int j) 是輔助方法,用于交換兩個(gè)數(shù)
  • void shellSort(int[] nums) 是算法主體
    • for (int gap = nums.length / 2; gap > 0; gap /= 2) 最外層循環(huán)踪区,設(shè)置每一次大循環(huán)的步長昆烁,更新時(shí)砍掉一半,直到步長為0
    • for (int i = gap; i < nums.length; i++) 第二層循環(huán)結(jié)合 for (int j = i; j >= gap; j -= gap) 第三層循環(huán)就是對每個(gè)分組進(jìn)行插入排序
      • 比如還是這個(gè)數(shù)組{4, 3, 5, 7, 6, 8, 9, 1, 2}缎岗,i = gap(i = 4), j = i = 4静尼,此時(shí)就是對第一個(gè)分組{4, 6, 2}進(jìn)行插入排序的第一次,通過判段if (nums[j] < nums[j - gap])也就是比較第一個(gè)組的46传泊,如果后者小鼠渺,就交換兩者位置swap(nums, j, j - gap),否則跳出最內(nèi)層循環(huán)眷细,因?yàn)榇藭r(shí)第一個(gè)分組的第一次插入排序已經(jīng)完成
      • 然后i++ == 5, j = i = 5拦盹,也就是對第二個(gè)分組{3, 8}進(jìn)行插入排序,后面i更新后執(zhí)行的操作以此類推
      • 等到第二層循環(huán)的i = 8溪椎,第三層j = i = 8普舆,則是第一組接著第一次插入排序后的第二次,這時(shí)會比較26校读,發(fā)現(xiàn)兩者需要交換沼侣,交換后再比較42發(fā)現(xiàn)又得交換,然后第一分組的插入排序就完成了
測試

用隨機(jī)生成的數(shù)據(jù)進(jìn)行測試歉秫,取平均值
4萬個(gè)數(shù)據(jù)蛾洛,用時(shí)9-13毫秒
40萬個(gè)數(shù)據(jù),用時(shí)90-110毫秒
400萬個(gè)數(shù)據(jù)雁芙,用時(shí)1300-1500毫秒
4000萬個(gè)數(shù)據(jù)轧膘,用時(shí)16000-20000毫秒(16-20秒)

而插入排序
4萬個(gè)數(shù)據(jù),用時(shí)900-1000毫秒
40萬個(gè)數(shù)據(jù)却特,用時(shí)76000-100000毫秒(70多到100秒)
數(shù)量級再往上都沒那個(gè)耐心等它跑完了

明顯看到希爾排序的效率是高很多很多

再測試了一下快速排序扶供,堆排序這兩種平均時(shí)間復(fù)雜度是O(n*logn)的排序算法筛圆,發(fā)現(xiàn)希爾排序與這兩種算法用時(shí)幾乎是一樣的裂明,不過希爾排序的平均時(shí)間復(fù)雜度至今仍不能給出確定的答案(O(n*logn), O(n^(1.3)), O(n^(1.5))),所以就先不管了

---------------------------華麗的分割線---------------------------------

用類似于上篇文章 插入排序 后面所講的改進(jìn),將希爾排序修改為以下實(shí)現(xiàn)闽晦,性能提高到原來的140% - 200%

    private static void shellSort(int[] nums) {
        for (int gap = nums.length / 2; gap > 0; gap /= 2) {
            for (int i = gap; i < nums.length; i++) {
                int insertValue = nums[i];
                int insertIndex = i - gap;
                while (insertIndex >= 0 && nums[insertIndex] > insertValue) {
                    nums[insertIndex + gap] = nums[insertIndex];
                    insertIndex -= gap;
                }
                if (insertIndex != i - gap) {
                    nums[insertIndex + gap] = insertValue;
                }
            }
        }
    }
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末扳碍,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子仙蛉,更是在濱河造成了極大的恐慌笋敞,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,277評論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件荠瘪,死亡現(xiàn)場離奇詭異夯巷,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)哀墓,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,689評論 3 393
  • 文/潘曉璐 我一進(jìn)店門趁餐,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人篮绰,你說我怎么就攤上這事后雷。” “怎么了吠各?”我有些...
    開封第一講書人閱讀 163,624評論 0 353
  • 文/不壞的土叔 我叫張陵臀突,是天一觀的道長。 經(jīng)常有香客問我贾漏,道長候学,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,356評論 1 293
  • 正文 為了忘掉前任纵散,我火速辦了婚禮盒齿,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘困食。我一直安慰自己边翁,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,402評論 6 392
  • 文/花漫 我一把揭開白布硕盹。 她就那樣靜靜地躺著符匾,像睡著了一般。 火紅的嫁衣襯著肌膚如雪瘩例。 梳的紋絲不亂的頭發(fā)上啊胶,一...
    開封第一講書人閱讀 51,292評論 1 301
  • 那天,我揣著相機(jī)與錄音垛贤,去河邊找鬼焰坪。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的超陆。 我是一名探鬼主播,決...
    沈念sama閱讀 40,135評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼黄刚,長吁一口氣:“原來是場噩夢啊……” “哼黔漂!你這毒婦竟也來了诫尽?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,992評論 0 275
  • 序言:老撾萬榮一對情侶失蹤炬守,失蹤者是張志新(化名)和其女友劉穎牧嫉,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體减途,經(jīng)...
    沈念sama閱讀 45,429評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡酣藻,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,636評論 3 334
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了鳍置。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片臊恋。...
    茶點(diǎn)故事閱讀 39,785評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖墓捻,靈堂內(nèi)的尸體忽然破棺而出抖仅,到底是詐尸還是另有隱情,我是刑警寧澤砖第,帶...
    沈念sama閱讀 35,492評論 5 345
  • 正文 年R本政府宣布撤卢,位于F島的核電站,受9級特大地震影響梧兼,放射性物質(zhì)發(fā)生泄漏放吩。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,092評論 3 328
  • 文/蒙蒙 一羽杰、第九天 我趴在偏房一處隱蔽的房頂上張望渡紫。 院中可真熱鬧,春花似錦考赛、人聲如沸惕澎。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,723評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽唧喉。三九已至,卻和暖如春忍抽,著一層夾襖步出監(jiān)牢的瞬間八孝,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,858評論 1 269
  • 我被黑心中介騙來泰國打工鸠项, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留干跛,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,891評論 2 370
  • 正文 我出身青樓祟绊,卻偏偏與公主長得像楼入,于是被迫代替她去往敵國和親哥捕。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,713評論 2 354

推薦閱讀更多精彩內(nèi)容

  • 轉(zhuǎn)載自:https://egoistk.github.io/2016/09/10/Java%E6%8E%92%E5...
    chad_it閱讀 987評論 0 18
  • 希爾排序 希爾排序是希爾(Donald Shell)于1959年提出的一種排序算法浅辙。希爾排序也是一種插入排序扭弧,它是...
    HuFan_JS閱讀 481評論 0 1
  • 排序算法幾種分類方式: 1阎姥,穩(wěn)定排序和不穩(wěn)定排序 如果a==b记舆, 當(dāng)排序之前a在b的前面,排序后呼巴,a仍然在b...
    fly_ever閱讀 416評論 0 0
  • 希爾排序個(gè)人感覺還是有一點(diǎn)難度的,當(dāng)時(shí)在理解的時(shí)候花了不少時(shí)間.希爾排序這里面有一個(gè)叫做"步長"的概念.就是每次通...
    maylor_zhu閱讀 334評論 0 0
  • 【日精進(jìn)打卡第253天】 【知~學(xué)習(xí)】 《六項(xiàng)精進(jìn)》5遍 共1089遍 《大學(xué)》5遍 共1084遍 【經(jīng)典名句分享...
    湯京潤0第361期0感謝三組閱讀 122評論 0 0