自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

各大排序算法性能比較及演示實例

開發(fā) 前端 算法
如果記錄中關(guān)鍵字不能重復(fù),則排序結(jié)果是唯一的,那么選擇的排序方法穩(wěn)定與否就無關(guān)緊要了;如果關(guān)鍵字可以重復(fù),則在選擇排序方法時,就要根據(jù)具體的需求來考慮選擇穩(wěn)定還是不穩(wěn)定的排序方法。那么,哪些排序算法是不穩(wěn)定的呢?

所謂排序,即將原來無序的一個序列重新排列成有序的序列。

排序方法中涉及到穩(wěn)定性,所謂穩(wěn)定性,是指待排序的序列中有兩個或兩個以上相同的項,在排序前和排序后看這些相同項的相對位置有沒有發(fā)生變化,如果沒有發(fā)生變化,即該排序方法是穩(wěn)定的,如果發(fā)生變化,則說明該排序方法是不穩(wěn)定的。

如果記錄中關(guān)鍵字不能重復(fù),則排序結(jié)果是唯一的,那么選擇的排序方法穩(wěn)定與否就無關(guān)緊要了;如果關(guān)鍵字可以重復(fù),則在選擇排序方法時,就要根據(jù)具體的需求來考慮選擇穩(wěn)定還是不穩(wěn)定的排序方法。那么,哪些排序算法是不穩(wěn)定的呢?

“快些選堆”:其中“快”指快速排序,“些”指希爾排序,“選”指選擇排序,“堆”指堆排序,即這四種排序方法是不穩(wěn)定的,其他自然都是穩(wěn)定的。

 

排序算法分類

1、插入類排序

即在一個已經(jīng)有序的序列中,插入一個新的記錄,就好比軍訓(xùn)排隊,已經(jīng)排好一個縱隊,這時來了個新家伙,于是新來的“插入”這個隊伍中的合適位置。這類排序有:直接插入排序、折半插入排序、希爾排序。

2、交換類排序

該類方法的核心是“交換”,即每趟排序,都是通過一系列的“交換”動作完成的,如軍訓(xùn)排隊時,教官說:你比旁邊的高,你倆交換下,還比下一個高就繼續(xù)交換。這類排序有:冒泡排序、快速排序。

3、選擇類排序

該方法的核心是“選擇”,即每趟排序都選出一個最?。ɑ?**)的記錄,把它和序列中的***個(或***一個)記錄交換,這樣最?。ɑ?**)的記錄到位。如軍訓(xùn)排隊時,教官選出個子最小的同學(xué),讓他和***個位置的同學(xué)交換,剩下的繼續(xù)選擇。這類排序有:選擇排序、堆排序。

4、歸并類排序

所謂歸并,就是將兩個或兩個以上的有序序列合并成一個新的有序序列。如軍訓(xùn)排隊時,教官說:每個人先和旁邊的人組成二人組,組內(nèi)排好隊,二人組和旁邊的二人組組成四人組,內(nèi)部再排好隊,以此類推,直到***全部同學(xué)都?xì)w并到一個組中并排好序。這類排序有:(二路)歸并排序。

5、基數(shù)類排序

此類方法較為特別,是基于多關(guān)鍵字排序的思想,把一個邏輯關(guān)鍵字拆分成多個關(guān)鍵字,如一副撲克牌,按照基數(shù)排序思想可以先按花色排序,則分成4堆,每堆再按A-K的順序排序,使得整副撲克牌最終有序。

排序算法分析

本文主要分析的排序算法有:冒泡排序、選擇排序、插入排序、希爾排序、快速排序、歸并排序、堆排序。

交換算法

由于大部分排序算法中使用到兩個記錄相互交換的動作,因此將交換動作單獨封裝出來,便于各排序算法使用。

  1. //交換函數(shù) 
  2. Array.prototype.swap = function(i, j) {  
  3.   var temp = this[i];  
  4.    this[i] = this[j];  
  5.    this[j] = temp;  

插入排序

算法思想:每趟將一個待排序的關(guān)鍵字,按照其關(guān)鍵字值的大小插入到已經(jīng)排好的部分序列的適當(dāng)位置上,直到插入完成。

  1. //插入排序 
  2. Array.prototype.insertionSort = function() {  
  3.     for (var i = 1; i < this.length; ++i)  
  4.     {  
  5.         var j = i, 
  6.             value = this[i];  
  7.         while (j > 0 && this[j - 1] > value)  
  8.         {  
  9.             this[j] = this[j - 1];  
  10.             --j;  
  11.         }  
  12.         this[j] = value;  
  13.     }  

算法性能:在內(nèi)層循環(huán)中this[j]=this[j-1],這句是作為基本操作??紤]最壞情況,即整個序列是逆序的,則其基本操作總的執(zhí)行次數(shù)為n*(n-1)/2,其時間復(fù)雜度為O(n*n)。考慮***情況,即整個序列已經(jīng)有序,則循環(huán)內(nèi)的操作均為常量級,其時間復(fù)雜度為O(n)。因此本算法平均時間復(fù)雜度為O(n*n)。算法所需的額外空間只有一個value,因此空間復(fù)雜度為O(1)。

希爾排序

算法思想:希爾排序又叫做縮小增量排序,是將待排序的序列按某種規(guī)則分成幾個子序列,分別對這幾個子序列進行插入排序,其中這一規(guī)則就是增量。如可以使用增量5、3、1來分格序列,且每一趟希爾排序的增量都是逐漸縮小的,希爾排序的每趟排序都會使得整個序列變得更加有序,等整個序列基本有序了,再使用一趟插入排序,這樣會更有效率,這就是希爾排序的思想。

  1. //希爾排序 
  2. Array.prototype.shellSort = function() {  
  3.     for (var step = this.length >> 1; step > 0; step >>= 1)  
  4.     {  
  5.         for (var i = 0; i < step; ++i)  
  6.         {  
  7.             for (var j = i + step; j < this.length; j += step)  
  8.             {  
  9.                 var k = j, value = this[j];  
  10.                 while (k >= step && this[k - step] > value)  
  11.                 {  
  12.                     this[k] = this[k - step];  
  13.                     k -= step;  
  14.                 }  
  15.                 this[k] = value;  
  16.             }  
  17.         }  
  18.     }  

算法性能:希爾排序的時間復(fù)雜度平均情況為O(nlogn),空間復(fù)雜度為O(1)。希爾排序的增量取法要注意,首先增量序列的***一個值一定是1,其次增量序列中的值沒有除1之外的公因子,如8,4,2,1這樣的序列就不要?。ㄓ泄蜃?)。

冒泡排序

算法思想:通過一系列的“交換”動作完成的,首先***個記錄與第二個記錄比較,如果***個大,則二者交換,否則不交換;然后第二個記錄和第三個記錄比較,如果第二個大,則二者交換,否則不交換,以此類推,最終***的那個記錄被交換到了***,一趟冒泡排序完成。在這個過程中,大的記錄就像一塊石頭一樣沉底,小的記錄逐漸向上浮動。冒泡排序算法結(jié)束的條件是一趟排序沒有發(fā)生元素交換。

  1. //冒泡排序 
  2. Array.prototype.bubbleSort = function() {  
  3.     for (var i = this.length - 1; i > 0; --i)  
  4.     {  
  5.         for (var j = 0; j < i; ++j)  
  6.             if (this[j] > this[j + 1]) 
  7.                 this.swap(j, j + 1);  
  8.     }  

算法性能:最內(nèi)層循環(huán)的元素交換操作是算法的基本操作。最壞情況,待排序列逆序,則基本操作的總執(zhí)行次數(shù)為(n-1+1)*(n-1)/2=n(n-1)/2,其時間復(fù)雜度為O(n*n);***情況,待排序列有序,則時間復(fù)雜度為O(n),因此平均情況下的時間復(fù)雜度為O(n*n)。算法的額外輔助空間只有一個用于交換的temp,所以空間復(fù)雜度為O(1)。

快速排序

算法思想:以軍訓(xùn)排隊為例,教官說以***個同學(xué)為中心,比他矮的站他左邊,比他高的站他右邊,這就是一趟快速排序。因此,一趟快速排序是以一個樞軸,將序列分成兩部分,樞軸的一邊比它?。ɑ蛐∮诘扔冢硪贿叡人螅ɑ虼笥诘扔冢?。

  1. //遞歸快速排序 
  2. Array.prototype.quickSort = function(s, e) {  
  3.     if (s == null
  4.         s = 0;  
  5.     if (e == null
  6.         e = this.length - 1;  
  7.     if (s >= e) 
  8.         return;  
  9.     this.swap((s + e) >> 1, e);  
  10.     var index = s - 1;  
  11.     for (var i = s; i <= e; ++i)   
  12.         if (this[i] <= this[e]) this.swap(i, ++index);  
  13.     this.quickSort(s, index - 1);  
  14.     this.quickSort(index + 1, e);  

算法性能:快速排序***情況下時間復(fù)雜度為O(nlogn),待排序列越接近無序,則該算法效率越高,在最壞情況下時間復(fù)雜度為O(n*n),待排序列越接近有序,則該算法效率越低,算法的平均時間復(fù)雜度為O(nlogn)。就平均時間而言,快速排序是所有排序算法中***的。該算法的空間復(fù)雜度為O(logn),快速排序是遞歸進行的,需要棧的輔助,因此需要的輔助空間比前面幾類排序方法要多。

快速排序的效率和選取的“樞軸”有關(guān),選取的樞軸越接近中間值,算法效率就越高,因此為了提高算法效率,可以在***次選取“樞軸”時做文章,如在數(shù)據(jù)堆中隨機選取3個值,取3個值的平均值作為“樞軸”,就如抽樣一般。關(guān)于具體如何提高快速排序算法的效率,在本文不做詳細(xì)介紹了,點到為止。(感興趣的讀者可以自行去研究)

選擇排序

算法思想:該算法的主要動作就是“選擇”,采用簡單的選擇方式,從頭至尾順序掃描序列,找出最小的一個記錄,和***個記錄交換,接著從剩下的記錄中繼續(xù)這種選擇和交換,最終使序列有序。

  1. //選擇排序 
  2. Array.prototype.selectionSort = function() {  
  3.     for (var i = 0; i < this.length; ++i)  
  4.     {  
  5.         var index = i;  
  6.         for (var j = i + 1; j < this.length; ++j)  
  7.         {  
  8.             if (this[j] < this[index]) 
  9.                 index = j;  
  10.         }  
  11.         this.swap(i, index);  
  12.     }  

算法性能:將最內(nèi)層循環(huán)中的比較視為基本操作,其執(zhí)行次數(shù)為(n-1+1)*(n-1)/2=n(n-1)/2,其時間復(fù)雜度為O(n*n),本算法的額外空間只有一個temp,因此空間復(fù)雜度為O(1)。

堆排序

算法思想:堆是一種數(shù)據(jù)結(jié)構(gòu),***的理解堆的方式就是把堆看成一棵完全二叉樹,這個完全二叉樹滿足任何一個非葉節(jié)點的值,都不大于(或不小于)其左右孩子節(jié)點的值。若父親大孩子小,則這樣的堆叫做大頂堆;若父親小孩子大,這樣的堆叫做小頂堆。根據(jù)堆的定義,其根節(jié)點的值是***(或最小),因此將一個無序序列調(diào)整為一個堆,就可以找出這個序列的***(或最?。┲担缓髮⒄页龅倪@個值交換到序列的***(或最前),這樣有序序列元素增加1個,無序序列中元素減少1個,對新的無序序列重復(fù)這樣的操作,就實現(xiàn)了序列排序。堆排序中最關(guān)鍵的操作是將序列調(diào)整為堆,整個排序的過程就是通過不斷調(diào)整使得不符合堆定義的完全二叉樹變?yōu)榉隙讯x的完全二叉樹的過程。

堆排序執(zhí)行過程(大頂堆):

(1)從無序序列所確定的完全二叉樹的***個非葉子節(jié)點開始,從右至左,從下至上,對每個節(jié)點進行調(diào)整,最終將得到一個大頂堆。將當(dāng)前節(jié)點(a)的值與其孩子節(jié)點進行比較,如果存在大于a值的孩子節(jié)點,則從中選出***的一個與a交換。當(dāng)a來到下一層的時候重復(fù)上述過程,直到a的孩子節(jié)點值都小于a的值為止。

(2)將當(dāng)前無序序列中***個元素,在樹中是根節(jié)點(a)與無序序列中***一個元素(b)交換。a進入有序序列,到達(dá)最終位置,無序序列中元素減少1個,有序序列中元素增加1個,此時只有節(jié)點b可能不滿足堆的定義,對其進行調(diào)整。

(3)重復(fù)過程2,直到無序序列中的元素剩下1個時排序結(jié)束。

  1. //堆排序 
  2. Array.prototype.heapSort = function() {  
  3.     for (var i = 1; i < this.length; ++i)  
  4.         {  
  5.         for (var j = i, k = (j - 1) >> 1; k >= 0; j = k, k = (k - 1) >> 1)  
  6.         {  
  7.             if (this[k] >= this[j]) 
  8.                 break;  
  9.             this.swap(j, k);  
  10.         }  
  11.     }  
  12.     for (var i = this.length - 1; i > 0; --i)  
  13.     {  
  14.         this.swap(0, i);  
  15.         for (var j = 0, k = (j + 1) << 1; k <= i; j = k, k = (k + 1) << 1)  
  16.         {  
  17.             if (k == i || this[k] < this[k - 1]) 
  18.                 --k;  
  19.             if (this[k] <= this[j]) 
  20.                 break;  
  21.             this.swap(j, k);  
  22.         }  
  23.     }  

算法性能:完全二叉樹的高度為[log(n+1)],即對每個節(jié)點調(diào)整的時間復(fù)雜度為O(logn),基本操作總次數(shù)是兩個并列循環(huán)中基本操作次數(shù)相加,則整個算法時間復(fù)雜度為O(logn)*n/2+O(logn)*(n-1),即O(nlogn)。額外空間只有一個temp,因此空間復(fù)雜度為O(1)。

堆排序的優(yōu)點是適合記錄數(shù)很多的場景,如從1000000個記錄中選出前10個最小的,這種情況用堆排序***,如果記錄數(shù)較少,則不提倡使用堆排序。另外,Hash表+堆排序是處理海量數(shù)據(jù)的***組合,關(guān)于海量數(shù)據(jù)處理會在之后的博文中介紹到。

歸并排序

算法思想:其核心就是“兩兩歸并”,首先將原始序列看成每個只含有單獨1個元素的子序列,兩兩歸并,形成若干有序二元組,則***趟歸并排序結(jié)束,再將這個序列看成若干個二元組子序列,繼續(xù)兩兩歸并,形成若干有序四元組,則第二趟歸并排序結(jié)束,以此類推,***只有兩個子序列,再進行一次歸并,即完成整個歸并排序。

  1. //歸并排序 
  2. Array.prototype.mergeSort = function(s, e, b) {  
  3.     if (s == null
  4.         s = 0;  
  5.     if (e == null
  6.         e = this.length - 1;  
  7.     if (b == null
  8.         b = new Array(this.length);  
  9.     if (s >= e) 
  10.         return;  
  11.     var m = (s + e) >> 1;  
  12.     this.mergeSort(s, m, b);  
  13.     this.mergeSort(m + 1, e, b);  
  14.     for (var i = s, j = s, k = m + 1; i <= e; ++i)   
  15.         b[i] = this[(k > e || j <= m && this[j] < this[k]) ? j++ : k++];  
  16.     for (var i = s; i <= e; ++i) 
  17.         this[i] = b[i];  

算法性能:可以選取“歸并操作”作為基本操作,“歸并操作”即為將待歸并表中元素復(fù)制到一個存儲歸并結(jié)果的表中的過程,其次數(shù)為要歸并的兩個子序列中元素個數(shù)之和。算法總共需要進行l(wèi)ogn趟排序,每趟排序執(zhí)行n次基本操作,因此整個歸并排序中總的基本操作執(zhí)行次數(shù)為nlogn,即時間復(fù)雜度為O(nlogn),說明歸并排序時間復(fù)雜度和初始序列無關(guān)。由于歸并排序需要轉(zhuǎn)存整個待排序列,因此空間復(fù)雜度為O(n)。

一些結(jié)論

(1)快速排序、希爾排序、歸并排序、堆排序的平均時間為O(nlogn),其他的為O(n*n)。

(2)快速排序、希爾排序、選擇排序、堆排序不穩(wěn)定,其他的穩(wěn)定。

(3)經(jīng)過一趟排序能夠保證一個元素到達(dá)最終位置的是冒泡排序、快速排序、選擇排序、堆排序。

(4)元素比較次數(shù)和原始序列無關(guān)的是選擇排序、折半插入排序。

(5)排序趟數(shù)和原始序列有關(guān)的是交換類排序。

(6)直接插入排序和折半插入排序的區(qū)別是尋找插入位置的方式不同,一個是按順序查找方式,另一個是按折半查找方式。

責(zé)任編輯:張燕妮 來源: twobin
相關(guān)推薦

2016-11-21 11:59:19

排序算法Objective-C

2017-12-14 10:16:01

CaddySSLDockerNginx

2020-09-14 14:47:13

排序算法

2011-04-15 10:26:38

JavaMVC

2009-05-25 08:39:08

iPhone蘋果移動OS

2011-05-18 14:52:04

XML

2020-10-23 07:43:37

Log配置性能

2013-12-16 10:20:48

MySQL數(shù)據(jù)庫

2023-11-20 10:34:09

語言

2009-12-04 19:28:25

FreeBSD 8.0Ubuntu 9.10性能比較

2010-12-27 16:01:45

jQuery選擇器

2012-12-03 10:26:51

Scala

2023-11-19 21:17:58

GoRust

2013-12-18 10:27:11

OpenMP線程

2011-08-09 10:21:55

SQL Server存儲過程分頁

2025-02-12 10:36:06

2009-07-01 18:12:18

JSP的優(yōu)勢性能比較

2011-07-19 09:35:46

.Net

2017-11-07 09:42:32

數(shù)據(jù)庫MySQLMEMORY

2010-03-10 16:35:23

Python編程語言
點贊
收藏

51CTO技術(shù)棧公眾號