Java(.NET)經典排序算法之快速排序

一、算法思想
     快速排序是C.R.A.Hoare於1962年提出的一種劃分交換排序。它採用了一種分治的策略,通常稱其爲分治法(Divide-and-ConquerMethod)。


(1) 分治法的基本思想
     分治法的基本思想是:將原問題分解爲若干個規模更小但結構與原問題相似的子問題。遞歸地解這些子問題,然後將這些子問題的解組合爲原問題的解。


(2)快速排序的基本思想
     設當前待排序的無序區爲R[low..high],利用分治法可將快速排序的基本思想描述爲:
①分解: 
     在R[low..high]中任選一個記錄作爲基準(Pivot),以此基準將當前無序區劃分爲左、右兩個較小的子區間R[low..pivotpos-1)和R[pivotpos+1..high],並使左邊子區間中所有記錄的關鍵字均小於等於基準記錄(不妨記爲pivot)的關鍵字pivot.key,右邊的子區間中所有記錄的關鍵字均大於等於pivot.key,而基準記錄pivot則位於正確的位置
(pivotpos)上,它無須參加後續的排序。
  注意:
     劃分的關鍵是要求出基準記錄所在的位置pivotpos。劃分的結果可以簡單地表示爲(注意pivot=R[pivotpos]):
      R[low..pivotpos-1].keys≤R[pivotpos].key≤R[pivotpos+1..high].keys
                  其中low≤pivotpos≤high。


②求解: 

     通過遞歸調用快速排序對左、右子區間R[low..pivotpos-1]和R[pivotpos+1..high]快速排序。


③組合: 

     因爲當"求解"步驟中的兩個遞歸調用結束時,其左、右兩個子區間已有序。對快速排序而言,"組合"步驟無須做什麼,可看作是空操作。


二、快速排序算法QuickSort

  1. void QuickSort(SeqList R,int low,int high)  
  2.  { //對R[low..high]快速排序  
  3.    int pivotpos; //劃分後的基準記錄的位置  
  4.    if(low<high){//僅當區間長度大於1時才須排序  
  5.       pivotpos=Partition(R,low,high); //對R[low..high]做劃分  
  6.       QuickSort(R,low,pivotpos-1); //對左區間遞歸排序  
  7.       QuickSort(R,pivotpos+1,high); //對右區間遞歸排序  
  8.     }  
  9.   } //QuickSort  


  注意:
     爲排序整個文件,只須調用QuickSort(R,1,n)即可完成對R[l..n]的排序。


三、劃分算法Partition
(1)簡單的劃分方法

具體做法
  第一步:(初始化)設置兩個指針i和j,它們的初值分別爲區間的下界和上界,即i=low,i=high;選取無序區的第一個記錄R[i](即R[low])作爲基準記錄,並將它保存在變量pivot中;
  第二步:令j自high起向左掃描,直到找到第1個關鍵字小於pivot.key的記錄R[j],將R[j])移至i所指的位置上,這相當於R[j]和基準R[i](即pivot)進行了交換,使關鍵字小於基準關鍵字pivot.key的記錄移到了基準的左邊,交換後R[j]中相當於是pivot;然後,令i指針自i+1位置開始向右掃描,直至找到第1個關鍵字大於pivot.key的記錄R[i],將R[i]移到i所指的位置上,這相當於交換了R[i]和基準R[j],使關鍵字大於基準關鍵字的記錄移到了基準的右邊,交換後R[i]中又相當於存放了pivot;接着令指針j自位置j-1開始向左掃描,如此交替改變掃描方向,從兩端各自往中間靠攏,直至i=j時,i便是基準pivot最終的位置,將pivot放在此位置上就完成了一次劃分。


②一次劃分過程
     一次劃分過程中,具體變化情況【參見動畫演示

( http://student.zjzk.cn/course_ware/data_structure/web/flashhtml/kuaisupaixu.htm)

③劃分算法:
 

  1. int Partition(SeqList R,int i,int j)  
  2.    {//調用Partition(R,low,high)時,對R[low..high]做劃分,  
  3.     //並返回基準記錄的位置  
  4.      ReceType pivot=R[i]; //用區間的第1個記錄作爲基準 '  
  5.      while(i<j){ //從區間兩端交替向中間掃描,直至i=j爲止  
  6.        while(i<j&&R[j].key>=pivot.key) //pivot相當於在位置i上  
  7.          j--; //從右向左掃描,查找第1個關鍵字小於pivot.key的記錄R[j]  
  8.        if(i<j) //表示找到的R[j]的關鍵字<pivot.key  
  9.            R[i++]=R[j]; //相當於交換R[i]和R[j],交換後i指針加1  
  10.        while(i<j&&R[i].key<=pivot.key) //pivot相當於在位置j上  
  11.            i++; //從左向右掃描,查找第1個關鍵字大於pivot.key的記錄R[i]  
  12.        if(i<j) //表示找到了R[i],使R[i].key>pivot.key  
  13.            R[j--]=R[i]; //相當於交換R[i]和R[j],交換後j指針減1  
  14.       } //endwhile  
  15.      R[i]=pivot; //基準記錄已被最後定位  
  16.      return i;  
  17.    } //partition  


四、快速排序執行過程
     快速排序執行的全過程可用遞歸樹來描述。





分析: 
     (1)遞歸執行的路線如圖中帶箭頭的包絡線所示。
     (2) 遞歸樹上每一結點左旁方括號表示當前待排序的區間,結點內的關鍵字是劃分的基準關鍵字
  注意:
     葉結點對應的子區間只有一個關鍵字,無須劃分,故葉結點內沒有基準關鍵字
  (3) 劃分後得到的左、右兩個子區間分別標在該結點的左、右兩個孩子結點的左邊方括號內。
【例】根結點左旁方括號[49,38,65,97,76,13,27,49]表示初始待排序的關鍵字,根內的49表示所選的劃分基準記錄的關鍵字,劃分結果是[27,28,13]49[76,97,65,49_],其左右子區間分別標在根結點的兩個孩子的左邊。
     (4) 每個分支結點右旁圓括號中的內容表示對該結點左旁區間的排序過程結束之後返回的結果。它是其左右孩子對應的區間排序完成之後,將左右孩子對應的排序結果分別放在該分支結點的關鍵字前後所得到的關鍵字序列。
【例】分支結點76的左右孩子對應的區間排序後的結果分別是(49_,65)和(97),將它們分別放在76的前後即得(49,65,76,97),這是對結點76左旁區間[76,97,,65,49]排序的結果。
     (5) 算法的執行順序是遞歸樹中的箭頭順序,實際上當把劃分操作視爲訪問結點的操作時,快速排序的執行過程相當於是先序遍歷其遞歸樹。
  注意:
     任何遞歸算法均可用遞歸樹來描述其執行過程。


五、快速排序各次劃分後的狀態變化
[49 38 65 97 76 13 27 49] //初始關鍵字
[27 38 13] 49 [76 97 65 49] //第1次劃分完成之後,對應遞歸樹第2層
[13] 27 [38] 49 [49 65] 76 [97] //對上一層各無序區劃分完成後,對應遞歸樹第3層
13 27 38 49 49 [65] 76 97 //對上一層各無序區劃分完成後,對應遞歸樹第4層
13 27 38 49 49 65 76 97 //最後的排序結果




六、算法分析
     快速排序的時間主要耗費在劃分操作上,對長度爲k的區間進行劃分,共需k-1次關鍵字的比較。


(1)最壞時間複雜度
     最壞情況是每次劃分選取的基準都是當前無序區中關鍵字最小(或最大)的記錄,劃分的結果是基準左邊的子區間爲空(或右邊的子區間爲空),而劃分所得的另一個非空的子區間中記錄數目,僅僅比劃分前的無序區中記錄個數減少一個。
     因此,快速排序必須做n-1次劃分,第i次劃分開始時區間長度爲n-i+1,所需的比較次數爲n-i(1≤i≤n-1),故總的比較次數達到最大值:
               Cmax = n(n-1)/2=O(n2)
     如果按上面給出的劃分算法,每次取當前無序區的第1個記錄爲基準,那麼當文件的記錄已按遞增序(或遞減序)排列時,每次劃分所取的基準就是當前無序區中關鍵字最小(或最大)的記錄,則快速排序所需的比較次數反而最多。


(2)最好時間複雜度
     在最好情況下,每次劃分所取的基準都是當前無序區的"中值"記錄,劃分的結果是基準的左、右兩個無序子區間的長度大致相等。總的關鍵字比較次數:
        0(nlgn)
注意:
     用遞歸樹來分析最好情況下的比較次數更簡單。因爲每次劃分後左、右子區間長度大致相等,故遞歸樹的高度爲O(lgn),而遞歸樹每一層上各結點所對應的劃分過程中所需要的關鍵字比較次數總和不超過n,故整個排序過程所需要的關鍵字比較總次數C(n)=O(nlgn)。
     因爲快速排序的記錄移動次數不大於比較的次數,所以快速排序的最壞時間複雜度應爲0(n2),最好時間複雜度爲O(nlgn)。


(3)基準關鍵字的選取
     在當前無序區中選取劃分的基準關鍵字是決定算法性能的關鍵。
  ①"三者取中"的規則
     "三者取中"規則,即在當前區間裏,將該區間首、尾和中間位置上的關鍵字比較,取三者之中值所對應的記錄作爲基準,在劃分開始前將該基準記錄和該區伺的第1個記錄進行交換,此後的劃分過程與上面所給的Partition算法完全相同。
  ②取位於low和high之間的隨機數k(low≤k≤high),用R[k]作爲基準
     選取基準最好的方法是用一個隨機函數產生一個取位於low和high之間的隨機數k(low≤k≤high),用R[k]作爲基準,這相當於強迫R[low..high]中的記錄是隨機分佈的。用此方法所得到的快速排序一般稱爲隨機的快速排序。
注意:
     隨機化的快速排序與一般的快速排序算法差別很小。但隨機化後,算法的性能大大地提高了,尤其是對初始有序的文件,一般不可能導致最壞情況的發生。算法的隨機化不僅僅適用於快速排序,也適用於其它需要數據隨機分佈的算法。


(4)平均時間複雜度
     儘管快速排序的最壞時間爲O(n2),但就平均性能而言,它是基於關鍵字比較的內部排序算法中速度最快者,快速排序亦因此而得名。它的平均時間複雜度爲O(nlgn)。


(5)空間複雜度
     快速排序在系統內部需要一個棧來實現遞歸。若每次劃分較爲均勻,則其遞歸樹的高度爲O(lgn),故遞歸後需棧空間爲O(lgn)。最壞情況下,遞歸樹的高度爲O(n),所需的棧空間爲O(n)。


(6)穩定性
     快速排序是非穩定的,例如[2,2,1]。



七、代碼實現

[java] view plaincopy
  1. /** 
  2.  * 一趟快速排序的算法是:  
  3.         1)、設置兩個變量I、J,排序開始的時候I:=1,J:=N;  
  4.         2)以第一個數組元素作爲關鍵數據,賦值給X,即X:=A[1];  
  5.         3)、從J開始向前搜索,即由後開始向前搜索(J:=J-1),找到第一個小於X的值,兩者交換;  
  6.         4)、從I開始向後搜索,即由前開始向後搜索(I:=I+1),找到第一個大於X的值,兩者交換;  
  7.         5)、重複第3、4步,直到I=J;  
  8.  */  
  9. public class QuickSort {  
  10.     public static void main(String[] args) {  
  11.         int[] source = { 49386597761327};  
  12.         System.out.print("初始關鍵字:");  
  13.         printArray(source);  
  14.         System.out.println("");  
  15.           
  16.         quickSort(source, 0, source.length - 1);  
  17.           
  18.         System.out.print("\n\n排序後結果:");  
  19.         printArray(source);  
  20.     }  
  21.   
  22.     /* 
  23.      * 先按照數組爲數據原型寫出算法,再寫出擴展性算法。數組{49,38,65,97,76,13,27} 
  24.      */  
  25.     public static void quickSort(int[] source, int low, int high) {  
  26.         int pivotPos;// 劃分後的基準記錄的位置  
  27.         if (low < high) {  
  28.             pivotPos = partition(source, low, high);//對source數組做劃分  
  29.             quickSort(source, low, pivotPos - 1);// 對左區間遞歸排序  
  30.             quickSort(source, pivotPos + 1, high);// 對右區間遞歸排序  
  31.         }  
  32.     }  
  33.   
  34.     public static int partition(int[] source, int low, int high) {  
  35.         int pivot = source[low];// 用區間的第1個記錄作爲基準  
  36.         while (low < high) {// 從區間兩端交替向中間掃描,直至low=high爲止  
  37.             while (low < high && source[high] >= pivot) { // 高位找到比povite大,則符合要求,繼續尋找    
  38.                 high--;//從右向左掃描,查找第1個關鍵字小於pivot的記錄source[j]  
  39.             }  
  40.             if (low < high) {//表示找到的source[j]的關鍵字<pivot  
  41.                 source[low++] = source[high];// 相當於交換source[i]和source[j],交換後i加1  
  42.             }  
  43.             System.out.print("從右向左掃描<----"+"  這趟排序結果:");  
  44.             printArray(source);  
  45.             while (low < high && source[low] <= pivot) {//低位開始找到比povite小,符合要求,繼續尋找    
  46.                 low++; //從左向右掃描,查找第1個關鍵字大於pivot的記錄source[i]  
  47.             }  
  48.             if (low<high) {//表示找到了source[i],使source[i]>pivot  
  49.                 source[high--]=source[low];//相當於交換source[i]和source[j],交換後j減1  
  50.             }  
  51.             System.out.print("從左向右掃描---->"+"  這趟排序結果:");  
  52.             printArray(source);  
  53.         }  
  54.         // 當low == high,完成一趟快速排序,此時low位相當於空,等待pivot補上  
  55.         source[low] = pivot;//基準記錄已被最後定位  
  56.           
  57.         System.out.print("pivot="+pivot+"          這趟排序結果:");  
  58.         printArray(source);  
  59.         System.out.println("\n");  
  60.         return low;  
  61.     }  
  62.       
  63.       
  64.     public static void printArray(int[] source) {  
  65.         for (int i = 0; i < source.length; i++) {  
  66.             System.out.print("\t" + source[i]);  
  67.         }  
  68.         System.out.println();  
  69.     }  
  70. }  




八、運行結果

[java] view plaincopy
  1. 初始關鍵字:  49  38  65  97  76  13  27  
  2.   
  3. 從右向左掃描<----  這趟排序結果: 27  38  65  97  76  13  27  
  4. 從左向右掃描---->  這趟排序結果: 27  38  65  97  76  13  65  
  5. 從右向左掃描<----  這趟排序結果: 27  38  13  97  76  13  65  
  6. 從左向右掃描---->  這趟排序結果: 27  38  13  97  76  97  65  
  7. 從右向左掃描<----  這趟排序結果: 27  38  13  97  76  97  65  
  8. 從左向右掃描---->  這趟排序結果: 27  38  13  97  76  97  65  
  9. pivot=49          這趟排序結果:   27  38  13  49  76  97  65  
  10.   
  11.   
  12. 從右向左掃描<----  這趟排序結果: 13  38  13  49  76  97  65  
  13. 從左向右掃描---->  這趟排序結果: 13  38  38  49  76  97  65  
  14. pivot=27          這趟排序結果:   13  27  38  49  76  97  65  
  15.   
  16.   
  17. 從右向左掃描<----  這趟排序結果: 13  27  38  49  65  97  65  
  18. 從左向右掃描---->  這趟排序結果: 13  27  38  49  65  97  97  
  19. pivot=76          這趟排序結果:   13  27  38  49  65  76  97  
  20.   
  21.   
  22.   
  23.   
  24. 排序後結果:  13  27  38  49  65  76  97  
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章