<>分而治之方法还可以用于实现另一种完全不同的排序方法,这种排序法称为快速排序(quick sort)。在这种方法中, n 个元素被分成三段(组):左段l e f t,右段r i g h t和中段m i d d l e。中段仅包含一个元素。左段中各元素都小于等于中段元素,右段中各元素都大于等于中段元素。因此l e f t和r i g h t中的元素可以独立排序,并且不必对l e f t和r i g h t的排序结果进行合并。m i d d l e中的元素被称为支点( p i v o t )。图1 4 - 9中给出了快速排序的伪代码。 ' c# I1 b/ Y9 {8 s 1 p3 k9 U1 F7 Y8 C! y7 d# Z0 i % `7 x" h& m- D; m z6 C2 e/ /使用快速排序方法对a[ 0 :n- 1 ]排序 $ D! c% m+ W2 |. Y* W . F$ [: }* Y7 `% m/ l从a[ 0 :n- 1 ]中选择一个元素作为m i d d l e,该元素为支点 8 v5 r4 ^4 V) _8 B) ]4 F! n9 O9 ` 9 {& h5 f" H0 ]把余下的元素分割为两段left 和r i g h t,使得l e f t中的元素都小于等于支点,而right 中的元素都大于等于支点 ; ?9 t3 X8 R7 m7 d& N4 A4 p: b
递归地使用快速排序方法对left 进行排序 ( @6 Z* [1 j' T% y6 _ 9 z. X' T- V8 W# G( i递归地使用快速排序方法对right 进行排序 ; |. a1 F: I! Y . O" i0 p5 ]# {所得结果为l e f t + m i d d l e + r i g h t ; M! t) d1 U! S" P8 ?7 G) S 0 i: z' I6 G0 K7 Q8 T( h# x图14-9 快速排序的伪代码 6 V# [0 P( T; {" B3 A ' Q+ @: k. K D. T/ r0 a$ Y: F6 A
考察元素序列[ 4 , 8 , 3 , 7 , 1 , 5 , 6 , 2 ]。假设选择元素6作为支点,则6位于m i d d l e;4,3,1,5,2位于l e f t;8,7位于r i g h t。当left 排好序后,所得结果为1,2,3,4,5;当r i g h t排好序后,所得结果为7,8。把right 中的元素放在支点元素之后, l e f t中的元素放在支点元素之前,即可得到最终的结果[ 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 ]。 " I1 [4 u: q2 j3 ?( E1 |% |( I6 J$ _: L# R1 k! E
把元素序列划分为l e f t、m i d d l e和r i g h t可以就地进行(见程序1 4 - 6)。在程序1 4 - 6中,支点总是取位置1中的元素。也可以采用其他选择方式来提高排序性能,本章稍后部分将给出这样一种选择。8 k% [5 q% m4 j+ q& s0 L
4 A1 a) K6 @4 ^# G
程序14-6 快速排序0 E5 X* g2 G' A* o' D
4 m6 n. ]8 g1 t) O
template<CLASS T>& A2 E. V* B# t! v6 m
- S n1 }: ]9 } U, V. l4 J7 Fvoid QuickSort(T*a, int n)4 l$ b3 O2 s- x) R7 \
: M G" `. |- w! w/ swhile (true) { " U- u8 \5 D/ X9 j # j9 \: v5 ^: i+ b+ v: v5 Xdo {// 在左侧寻找>= pivot 的元素 . P4 E% `) ]& K9 \1 w* [) {4 x4 n& B2 K& y& k/ @ ?. d8 r, [; q8 E
i = i + 1; 7 h# {/ o: n+ Q: N; {) H- k% g , k+ y9 w l* J} while (a < pivot);6 L0 A) Q5 X: \$ J0 R; @: d7 X
+ v6 w& a; U0 W/ M
do {// 在右侧寻找<= pivot 的元素/ X: e, H- V: u2 ]
7 K% t L2 s8 k* S0 y" B0 F
j = j - 1; , o. y3 [9 C* V+ ]0 ? | 5 a' T+ h1 P6 k; R% d6 f} while (a[j] > pivot); $ O* S& w6 O* V) e+ ?$ Y+ a! X3 Z1 P8 T5 [
if (i >= j) break; // 未发现交换对象4 W A6 O. b1 ]
, V, `9 {& H2 n5 B6 K' p# W
Swap(a, a[j]);& q7 |1 P7 K% A
. J% O4 V6 M- ^! _4 o
}5 A9 {' ]5 I/ V; c, C
! l% S% Z* d! l( u: }4 w. r
// 设置p i v o t b9 j( ^0 @4 R7 f) J5 W2 |) j# I# _
a[l] = a[j]; & Z( K6 D( H& |- G3 R' }+ b / a% A @( l7 V- Pa[j] = pivot; ( J8 i7 o. v9 F9 r! B* `4 a4 B2 ]+ M3 g: d1 z& x' n/ E8 V: G$ @, N c
quickSort(a, l, j-1); // 对左段排序 % Y, f0 }9 o3 J3 {% E, I; a2 K
quickSort(a, j+1, r); // 对右段排序 ' h& S/ n$ n4 o$ c 4 E* f4 \) ?8 Q6 v' S/ |}: M+ M( ]& A. \. f8 j" Q( u+ C
7 D) s E1 ?; {3 K/ {
若把程序1 4 - 6中d o - w h i l e条件内的<号和>号分别修改为< =和> =,程序1 4 - 6仍然正确。实验结果表明使用程序1 4 - 6的快速排序代码可以得到比较好的平均性能。为了消除程序中的递归,必须引入堆栈。不过,消除最后一个递归调用不须使用堆栈。消除递归调用的工作留作练习(练习1 3)。程序1 4 - 6所需要的递归栈空间为O (n)。若使用堆栈来模拟递归,则可以把这个空间减少为O ( l o gn)。在模拟过程中,首先对left 和right 中较小者进行排序,把较大者的边界放入堆栈中。在最坏情况下l e f t总是为空,快速排序所需的计算时间为(n2 )。在最好情况下, l e f t和r i g h t中的元素数目大致相同,快速排序的复杂性为(nl o gn)。令人吃惊的是,快速排序的平均复杂性也是(nl o gn)。 W5 ]# s# h4 U0 n 9 q: N F. y" ]8 L定理2-1 快速排序的平均复杂性为(nl o gn)。# ?# I8 M- R$ i9 {7 l" j. L6 i
. P4 F, U4 @9 x/ S7 q3 y
证明用t (n) 代表对含有n 个元素的数组进行排序的平均时间。当n≤1时,t (n)≤d,d为某一常数。当n <1时,用s 表示左段所含元素的个数。由于在中段中有一个支点元素,因此右段中元素的个数为n-s- 1。所以左段和右段的平均排序时间分别为t (s), t (n-s- 1 )。分割数组中元素所需要的时间用cn 表示,其中c 是一个常数。因为s 有同等机会取0 ~n- 1中的任何一个值.. x+ Y9 S s8 ]7 q) {& I
& j4 u9 j- g1 ?' P( U& [: L如对(2 - 8)式中的n 使用归纳法,可得到t (n)≤kn l o ge n,其中n> 1且k=2(c+d),e~2 . 7 1 8为自然对数的基底。在归纳开始时首先验证n= 2时公式的正确性。根据公式( 1 4 - 8),可以得到t( 2 )≤2c+ 2d≤k nl o ge 2。在归纳假设部分,假定t(n)≤kn l o ge n(当2≤n<m 时,m 是任意一个比2大的整数=. . k- w3 t6 O8 g6 s, G$ r- t) o* J ' W) t. k7 c7 g8 B4 C# k& a. m$ `图1 4 - 1 0对本书中所讨论的算法在平均条件下和最坏条件下的复杂性进行了比较。2 F6 \3 K! v, r" K; l
) S c& e9 A% G: p G& M: N, `, _4 L/ k
方法最坏复杂性平均复杂性 4 I5 r7 k X1 J/ H0 E& L8 \% A) a% \% Q1 Q; s
冒泡排序n2 n2 ; W) g" F2 G0 ? V) j" n/ I* z# [3 @5 C$ i$ s
计数排序n2 n2 $ h, x9 L) `0 n2 D U& M : G' }) a- h3 @: R) L/ C1 e; v插入排序n2 n2 5 Z8 M) P8 ~6 ?2 v! J$ V4 G m5 @8 y# J# q# `: L9 A9 e
选择排序n2 n2 6 k& `( t) ]1 ^7 F5 R9 Y0 x0 F1 }
堆排序nl o gn nl o gn 6 `1 c5 w: P" X! M# [: m! Y( k X4 ^: f9 ?- x `" ?
归并排序nl o gn nl o gn5 k& k- E8 w. \1 w+ Y" `
2 [$ J, u+ F% e% _% h
快速排序n2 nl o gn! t1 C* e5 O# `0 N7 h/ U! E
& |) P9 e" s7 q# ?图14-10 各种排序算法的比较 % {. `1 [) C* C / r8 Q4 r N7 f `9 P& Y7 l; G) j) N0 M2 e) X- D; E$ y0 Y' M
中值快速排序( median-of-three quick sort)是程序1 4 - 6的一种变化,这种算法有更好的平均性能。注意到在程序1 4 - 6中总是选择a [ 1 ]做为支点,而在这种快速排序算法中,可以不必使用a [ 1 ]做为支点,而是取{a[1],a[(1+r)/2],a[r]} 中大小居中的那个元素作为支点。例如,假如有三个元素,大小分别为5,9,7,那么取7为支点。为了实现中值快速排序算法,一种最简单的方式就是首先选出中值元素并与a[1] 进行交换,然后利用程序1 4 - 6完成排序。如果a [ r ]是被选出的中值元素,那么将a[1] 与a[r] 进行交换,然后将a [ 1 ](即原来的a [ r ])赋值给程序1 4 - 6中的变量p i v o t,之后继续执行程序1 4 - 6中的其余代码。 . Z5 c8 }; Q# b1 Y. X' R" Q3 o' B: b 1 `1 |2 U3 r. r/ F图2 - 11中分别给出了根据实验所得到的归并排序、堆排序、插入排序、快速排序的平均时间。对于每一个不同的n, 都随机产生了至少1 0 0组整数。随机整数的产生是通过反复调用s t d l i b . h库中的r a n d o m函数来实现的。如果对一组整数进行排序的时间少于1 0个时钟滴答,则继续对其他组整数进行排序,直到所用的时间不低于1 0个时钟滴答。在图2 - 11中的数据包含产生随机整数的时间。对于每一个n,在各种排序法中用于产生随机整数及其他开销的时间是相同的。因此,图2 - 11中的数据对于比较各种排序算法是很有用的。 6 b$ R$ C( A. P: G: b6 E1 }6 Y& t* M. t ?8 M9 f
对于足够大的n,快速排序算法要比其他算法效率更高。从图中可以看到快速排序曲线与插入排序曲线的交点横坐标比2 0略小,可通过实验来确定这个交点横坐标的精确值。可以分别用n = 1 5 , 1 6 , 1 7 , 1 8 , 1 9进行实验,以寻找精确的交点。令精确的交点横坐标为nBr e a k。当n≤nBreak 时,插入排序的平均性能最佳。当n>nBreak 时,快速排序性能最佳。当n>nBreak 时,把插入排序与快速排序组合为一个排序函数,可以提高快速排序的性能,实现方法是把程序1 4 - 6中的以下语句:% d# a/ W2 U$ D1 H8 _7 N) v
/ s2 X5 O! c1 a( ?2 J+ \
if(l >= r)r e t u r n ;$ u9 \3 s, n6 f; ~
1 N$ Z) h' Y% P m" c替换为 # ~5 U m7 f6 I: { / o" s1 Q2 k3 P3 ^% P( \if (r-1<NBREAK) {InsertionSort(a,l,r); return;} 0 G: |" r2 X, F( l$ x& m # D0 T- q8 y' O, N
这里I n s e r t i o n S o r t ( a , l , r )用来对a [ 1 : r ]进行插入排序。测量修改后的快速排序算法的性能留作练习(练习2 0)。用更小的值替换n B r e a k有可能使性能进一步提高(见练习2 0)。 . H/ L$ t5 k0 {5 o 9 Q4 P) O& d7 b! R' |大多数实验表明,当n>c时(c为某一常数),在最坏情况下归并排序的性能也是最佳的。而当n≤c时,在最坏情况下插入排序的性能最佳。通过将插入排序与归并排序混合使用,可以提高归并排序的性能(练习2 1)。</P>