希爾排序(上)

2021-06-08 17:15:27 字數 2274 閱讀 7704

9.6.1 變不可能為可能

給大家出一道智力題。請問「vii」是什麼?

嗯,很好,它是羅馬數字的7。現在我們要給它加上一筆,讓它變成8(viii),應該是非常簡單,只需要在右側加一豎線即可。

現在我請大家試著對羅馬數字9,也就是「ix」增加一筆,把它變成6,應該怎麼做?

(幾分鐘後)

我已經聽不少聲音說,「這怎麼可能!」 可為什麼一定要用常規方法呢?

我這裡有三種另類的方法可以實現它。

方法一:觀察發現「x」其實可以看作是乙個正放乙個倒置兩個「v」。因此我們,給「ix」中間加一條水平線,上下顛倒,然後遮住下面部分,也就是說,我們所謂的加上一筆就是遮住一部分,於是就得到「vi」。

方法二:在「ix」前面加乙個「s」,此時構成乙個英文單詞「six」,這就等於得到乙個6了。哈哈,我聽到下面一片譁然,我剛有沒有說一定要是「vi」呀,我只說把它變成6而已,至於是羅馬數字還是英文單詞,我可沒有限制。顯然,你們的思維受到了我前面舉例的「vii」轉變為「viii」的影響。

方法三:在「ix」後面加乙個「6」,得到「1x6」,其結果當然是數字6了。大家笑了,因為這個想法實在是過分,把字母「i」當成了數字1,字母「x」看成了乘號。可誰又規定說這是不可以的呢?只要沒違反規則,得到6即可。

智力題的答案介紹完了 。大家會發現,看似解決不了的問題,還真不一定就沒有辦法,也許只是暫時沒想到罷了。

我們都能理解,優秀排序演算法的首要條件就是速度(還有其他要求,速度是第一位的)。於是人們想了許許多多的辦法,目的就是為了提高排序的速度。而在很長的時間裡,眾人發現儘管各種排序演算法花樣繁多(比如前面我們提到的三種不同的排序演算法),但時間複雜度都是o(n2),似乎沒法超越了(這裡排序指內排序)。此時,計算機學術界充斥著「排序演算法不可能突破o(n2)」的聲音。就像剛才大家做智力題的感覺一樣,「不可能」成了主流。

終於有一天,當一位科學家發布超越了o(n2)新排序演算法後,緊接著就出現了好幾種可以超越o(n2)的排序演算法,並把內排序演算法的時間複雜度提公升到了o(nlog2n)。「不可能超越o(n2)」徹底成為了歷史。

從這裡也告訴我們。做任何事,你解決不了時,想一想「nothing is impossible!」,雖然有點唯心,但這樣思維方式會讓你更加深入的思考解決方案,而不是匆忙的放棄。

9.6.2 希爾排序原理

現在,我要講解的演算法叫希爾排序(shell sort)。希爾排序是d.l.shell於2023年提出來的一種排序演算法,在這之前排序演算法的時間複雜度基本都是o(n2)的,希爾排序演算法是突破這個時間複雜度的第一批演算法之一。

我們前一節講的直接插入排序,應該說,它的效率在某些時候是很高的,比如,我們的記錄本身就是基本有序的,我們只需要少量的插入操作,就可以完成整個記錄集的排序工作,此時直接插入很高效。還有就是記錄數比較少時,直接插入的優勢也比較明顯。可問題在於,兩個條件本身就過於苛刻,現實中記錄少或者基本有序都屬於特殊情況。

不過別急,有條件當然是好,條件不存在,我們創造條件,也是可以去做的。於是科學家希爾研究出了一種排序,對直接插入排序改進後可以增加效率的方法。

如何讓待排序的記錄個數較少呢?很容易想到的就是將原本有大量記錄數的記錄進行分組。分割成若干個子串行,此時每個子串行待排序的記錄個數就比較少了。然後在這些子串行內分別進行直接插入排序,當整個序列都基本有序時,注意只是基本有序時,再對全體記錄進行一次直接插入排序。

此時一定有同學開始疑惑了。這不對呀,比如我們現在有序列是,現在將它分成三組,,,,哪怕將它們各自排序排好了,變成,,,再合併它們成,此時,這個序列還是雜亂無序,談不上基本有序,要排序還是重來一遍直接插入有序,這樣做有用嗎?需要強調一下,所謂的基本有序,就是小的關鍵字基本在前面,大的基本在後面,不大不小的基本在中間,像這樣可以稱為基本有序了。但像這樣的9在第三位,2在倒數第三位就談不上基本有序。

問題其實也就在這裡,我們分割待排序記錄的目的是減少待排序記錄的個數,並使整個序列向基本有序發展。而如上面這樣分完組後,就各自排序的方法達不到我們的要求。因此,我們需要採取跳躍分割的策略:將相距某個「增量」的記錄組成乙個子串行,這樣才能保證在子串行內分別進行直接插入排序後得到的結果是基本有序而不是區域性有序

排序 上 插排 希爾 堆排

給每個新增的數在已有的數列中找到合適的位置,然後插入進去 例如乙個由小到大的排序 原陣列為 4,5,2,9,3,6,8,7 現在我們有了乙個陣列,我們可以假裝現在排過序的只有第乙個數,也就是說已有序列是第乙個數 i和j是下標,key代表當前要插入的數,j代表當前已經插入到了第幾個數,i是動態的用來與...

排序 希爾排序

希爾排序 shell sort 又稱為縮小增量排序,輸入插入排序演算法,是對直接排序演算法的一種改進。本文介紹希爾排序演算法。對於插入排序演算法來說,如果原來的資料就是有序的,那麼資料就不需要移動,而插入排序演算法的效率主要消耗在資料的移動中。因此可知 如果資料的本身就是有序的或者本身基本有序,那麼...

排序 希爾排序

縮小增量排序 待排序列按關鍵字基本有序時,直接插入排序的效率很高 希爾排序思想 將整個待排記錄分割為若干子串行分別進行直接插入排序,待整個序列中的記錄基本有序時,再對全體記錄進行一次直接插入排序,就可以完成整個排序工作 增量序列中的值沒有除1以外的公因子,且最後乙個增量必須等於1 子串行的構成不是逐...