快速排序-常见中轴(主元pivot)选择方法及实现代码(末位/随机/三数中值/..)

来源:互联网 发布:三鹿乳业的网络公关 编辑:程序博客网 时间:2024/05/01 12:22

一、选取最后一个元素

在我们的课本中,看到最多的就是选择第一个元素作为中轴,但是在很多书上却选择最后一个元素作为中轴。下面就让我们来一睹选取最后一个元素作为中轴的快排。


注:本文中的所有算法都采用双向扫描法,即,设两个下标i和j,i和右扫描,j向左扫描,直到i不小于j。而当下标为i的数小于中轴时,跳过并继续向右扫描,否则停止扫描,并开始j的向左扫描,相对地,当下标为j的数大于中轴时,跳过并继续向右扫描,否则停止扫描,然后交换下标为i和j的两个数,并从下一个位置继续两个方向的扫描,直到i不小于j。最后把中轴与下标为i的元素交换即完成一趟的快排。


下面就以最后一个元素作为中轴来说明。


初始数据为: 8  1  4  9  0  3  5  2  7  6
中轴为:6,第一趟快排的情景如下:

8  1  4  9  0  3  5  2  7  6

↑                               ↑

i→                         ←j
 第一次交换后:
 2  1  4  9  0  3  5  8  7  6
 ↑                          ↑
 i                            j

第二次交换后:
2  1  4  5  0  3  9  8  7  6
            ↑          ↑
            i           j
第三次交换前:
2  1  4  5  0  3  9  8  7  6
                    ↑  ↑
                     j   i
将下标为i的元素与最后一个元素(中轴)交换
     即,一趟快排后:
             2  1  4  5  0  3  6  8  7  9
对左边的数组和右边的数组重复上述过程。


选取最后一个元素作为中轴的好处是简单直观,操作一致。因为我们通常把下标为i的元素与中轴元素作为交换,而i则总是指向比中轴大的元素,而把大的元素放到后面总是合理的。


二、双向描述法中的越界问题

快速排序中都非常喜欢使用双向扫描法,然而这个方法却存在一个越界问题,考虑如下的情况:
          8  1  4  6  9  3  5  2  7  0
           i→                        ←j

当我们选取最后一个元素作为中轴时,j向左扫描,一直到找到一个比0小的数,但是它却是数组中的最小值,所以j会一直向左走,直到越界。


这个问题无论是使用第一个元素作为中轴,还是使用最后一个元素作为中轴都会存在。所以,在双向描述中,必然有一个方向(远离中轴的方法)要做越界检查。


选取最后一个元素作为中轴的快排的关键代码

[cpp] view plaincopyprint?
  1. void QSort(DataType *data, int left, int right)  
  2. {  
  3.     //如果数据的个小数为1或0则不需要排序  
  4.     if(left >= right)  
  5.         return;  
  6.     //取最后一个元素作为枢纽  
  7.     DataType centre = data[right];  
  8.     int i = left;  
  9.     int j = right-1;  
  10.     while(true)  
  11.     {  
  12.         //从前向后扫描,找到第一个小于枢纽的值,  
  13.         //在到达数组末尾前,必定结果循环,因为最后一个值为centre  
  14.         while(data[i] < centre)  
  15.             ++i;  
  16.         //从后向前扫描,此时要检查下标,防止数组越界  
  17.         while(j >= left && data[j] > centre)  
  18.             --j;  
  19.         //如果没有完成一趟交换,则交换  
  20.         if(i < j)  
  21.             Swap(data[i++], data[j--]);  
  22.         else  
  23.             break;  
  24.     }  
  25.     //把枢纽放在正确的位置  
  26.     Swap(data[i], data[right]);  
  27.     QSort(data, left, i - 1);  
  28.     QSort(data, i + 1, right);  
  29. }  

三、随机选元法

我们知道,在快排中,中轴的选择对于算法的效率是非常重要的,选择一个好的中轴选择策略会使算法的效率显著提高。


无论是前面说的选取第一个元素还是最后一个元素作为中轴,其实都是一个坏的选元方法。因为当元素基本有序时,这两种方法都会使算法的效率非常低,最坏情况下,是O(n^2)。


随机选元法的思路:使用随机数生成函数生成一个随机数rand,随机数的范围为[left, right],并用此随机数为下标对应的元素a[rand]作为中轴,并与最后一个元素a[right]交换,然后进行与选取最后一个元素作为中轴的快排一样的算法即可。


随机选元法仍然存在扫描越界问题,所以在远离中轴的方法上仍然需要检查下标。


随机选元法的关键代码

[cpp] view plaincopyprint?
  1. void QSort(DataType *data, int left, int right)  
  2. {  
  3.     //如果数据的个小数为1或0则不需要排序  
  4.     if(left >= right)  
  5.         return;  
  6.     //随机选取一个元素作为枢纽,并与最后一个元素交换  
  7.     int ic = Random(left, right);  
  8.     Swap(data[ic], data[right]);  
  9.   
  10.   
  11.     DataType centre = data[right];  
  12.     int i = left;  
  13.     int j = right-1;  
  14.     while(true)  
  15.     {  
  16.         //从前向后扫描,找到第一个小于枢纽的值,  
  17.         //在到达数组末尾前,必定结果循环,因为最后一个值为centre  
  18.         while(data[i] < centre)  
  19.             ++i;  
  20.         //从后向前扫描,此时要检查下标,防止数组越界  
  21.         while(j >= left && data[j] > centre)  
  22.             --j;  
  23.         //如果没有完成一趟交换,则交换  
  24.         if(i < j)  
  25.             Swap(data[i++], data[j--]);  
  26.         else  
  27.             break;  
  28.     }  
  29.     //把枢纽放在正确的位置  
  30.     Swap(data[i], data[right]);  
  31.     QSort(data, left, i - 1);  
  32.     QSort(data, i + 1, right);  
  33. }  
  34. inline int Random(int begin, int end)  
  35. {  
  36.     //产生begin至end,包括begin和end的随机数,即[begin, end]范围的整数  
  37.     return rand()%(end - begin + 1) + begin;  
  38. }  

从上面的代码也可以看出,随机选法与选取最后一个元素作为中轴的方法是非常相近的,只是多了把随机的中轴放到最后的位置的操作。


四、三数中值分割法

一组N个数的中值是第N/2个最大数,中轴的最好选择就是数组的中值。不幸的是,这很难算出。但这样的中值的估计量可以通过随机选取三个元素并用它们的中值作为中轴而得到。事实上,随机性并没有多大的帮助,因此一般的做法是使用左端、右端和中心位置上的三个元素的中值作为中轴。


分割策略:假设数组被排序的范围为left和right,center=(left+right)/2,对a[left]、a[right]和a[center]进行适当排序,取中值为中轴,将最小者放a[left],最大者放在a[right],把中轴元与a[right-1]交换,并在分割阶段将i和j初始化为left+1和right-2。然后使用双向描述法,进行快排。


三数中值分割法的实现

初始数据:                    6  1  8  9  4  3  5  2  7  0
对三个数进行排序后:   1  8  9  4  3  5  2  7  6
中轴与a[right-1]交换:  0  1  8  9  7  3  5  2  4  6
开始扫描:                         i→               ←j
第一次交换后:            0  1   9  7  3  5  8  4  6
                                              i =2              j=7
第二次交换后:            0  1  2  3  7  9  5  8  4  6
                                                  i=3   j=5
第三次交换前:            0  1  2  3  7  9  5  8  4  6
  i=4,j= 3                                3=j   i=4
第三次交换后:            0  1  2  3  4  9  5  8  7  6 
     (与a[right-1]交换) 


分割策略的好处

1)将三元素中最小者被分到a[left]、最大者分到a[right]是正确的,因为当快排一趟后,比中轴小的放到左边,而比中轴大的放到右边,这样就在分割的时候把它们分到了正确的位置,减少了一次比较和交换。
2)在前面所说的所有算法中,都有双向扫描时的越界问题,而使用这个分割策略则可以解决这个问题。因为i向右扫描时,必然会遇到不小于中轴的数a[right-1],而j在向左扫描时,必然会遇到不大于中轴的数a[left],这样,a[right-1]和a[left]提供了一个警戒标记,所以不需要检查下标越界的问题。

关键实现代码

[cpp] view plaincopyprint?
  1. DataType Median3(DataType *data, int left, int right)  
  2. {  
  3.     //取数据的头、尾和中间三个数,并对他们进行排序  
  4.     //排序结果直接保存在数组中  
  5.     int centre = (left + right)/2;  
  6.     if(data[left] > data[centre])  
  7.         Swap(data[left], data[centre]);  
  8.     if(data[left] > data[right])  
  9.         Swap(data[left], data[right]);  
  10.     if(data[centre] > data[right])  
  11.         Swap(data[centre], data[right]);  
  12.     //把中值,即枢纽与数组倒数第二个元素交换  
  13.     swap(data[centre], data[right - 1]);  
  14.     return data[right - 1];//返回枢纽  
  15. }  
  16.   
  17. void QSort(DataType *data, int left, int right)  
  18. {  
  19.     //如果需要排序的数据大于3个则使用快速排序  
  20.     if(right - left >= 3)  
  21.     {  
  22.         //取得枢纽的值  
  23.         DataType centre = Median3(data, left, right);  
  24.         int begin = left;  
  25.         int end = right - 1;  
  26.         while(true)  
  27.         {  
  28.             //向后扫描数组  
  29.             //由于在选择枢纽时,已经把比枢纽值大的数据放在right位置  
  30.             //所以不会越界  
  31.             while(data[++begin] < centre);  
  32.             //向前扫描数组  
  33.             //由于在选择枢纽时,已经把比枢纽值小的数据放在left位置  
  34.             //所以不会越界  
  35.             while(data[--end] > centre);  
  36.             //把比枢纽小的数据放在前部,大的放到后部  
  37.             if(begin < end)  
  38.                 Swap(data[begin], data[end]);  
  39.             else  
  40.             {  
  41.                 //已经对要排序的数据都与枢纽比较了一次  
  42.                 //把中枢纽保存在适当的位置,因为begin的数一定比枢纽大  
  43.                 //所以把这个数放在数组后面  
  44.                 Swap(data[begin], data[right - 1]);  
  45.                 break;  
  46.             }  
  47.         }  
  48.   
  49.   
  50.         QSort(data, left, begin - 1);  
  51.         QSort(data, begin + 1, right);  
  52.     }  
  53.     else //如果要排序的数据很少,少于等于3个,则直接使用冒泡排序  
  54.     {  
  55.         BubbleSort(data+left, right - left + 1);  
  56.     }  
  57. }  
注意当要排序的数据很少时(小于3个),则不能进行三数取中值,此时直接使用简单的排序(例如冒泡)即可,而且从效率的角度来考虑这也是合理的,因为可以避免函数调用的开销。

五、与中轴相等时的操作

当i或j在扫描中遇到与中轴相等的元素时,是停止扫描还是继续?


我们仍然采用双向扫描法,而且从实践中我们知道,i与j的操作应该相同(停止或继续)否则,一趟快排后,会出现中轴偏向一边的现象。


下面以一下极端的例子来说明:

采用三数中值分割法
初始数据:         2  2  2  2  2  2  2  2  2
分割后:             2  2  2  2  2  2  2  2  2
                                i→            ←j
1)继续扫描(先不考虑越界)   
                             2  2  2  2  2  2  2  2  2
                             j                            i
第一趟快排后: 2  2  2  2  2  2  2   2
    (a[i]与a[right-1]交换)


我们可以看到,如果继续扫描,出现的将是快排中的最坏情况,分出来的数组极其不平均,时间复杂度为O(N^2)。采用前面的两种快排算法也同样。


2)停止扫描并交换数据
初始数据:             2  2  2  2  2  2  2  2  2
分割后:                 2  2  2  2  2  2  2  2  2
                                    i→            ←j
第一次交换后:     2  2  2  2  2  2  2  2  2
                                   i=1               j=6
第二次交换后:     2  2  2  2  2  2  2  2  2
                                       i=2       j=5
第三次交换后:     2  2  2   2  2  2  2  2
                                      3= i   j=4
第四前交换后:     2  2  2  2  2  2  2  2  2
(与a[right-1]交换)     3= j   i=4


虽然停止扫描并交换两个数据看起来并没有什么意义,因为所有的数据都是相等的,但是它却可以把数组平均地分为两个子数组。使快排达到最理想的情况,也就是说,我们是用交换来换来了效率,真是不可思议。


同时,在上面的分析中,我们是忽略了下标越界的检查的,从上面的分析中,我们可以看到,如果继续扫描(第1种情况),则需要在两个方向上做检查(这里所讲的所有算法都需要),它与我们之前所说的算法都非常的不一致。更加让人无法容忍的是,它会使快排变成最坏的情况。


在第2种情况中,虽然要交换元素,但是,却使快排以最理想的情况运行,并使算法与之前描述的完全一致。
总之,当遇到与中轴相等的元素时,应停止扫描。所以在我们的所有代码中,while循环中的条件都没有“=”。


极端的情况并不极端

上面的例子中,举出了一个非常极端的例子来说明遇到与中轴相等的元素时的处理方法。其实在现实中,这种情况并不极端。


试想要排序的数据有10,000个,其中有2,000个元素相等,并随机分布在数组中,也就是说平均在一个有10个元素的数组中有2个元素相等,这是非常普遍的情况。然而随着快排的进行,这2,000个相等的数,最终会被交换到一个连续的空间中,并出现了前面我们所说的极端情况。它就会成为我们算法的一个瓶颈。

0 0
原创粉丝点击