<>可以运用分而治之方法来解决排序问题,该问题是将n 个元素排成非递减顺序。分而治之方法通常用以下的步骤来进行排序算法:若n 为1,算法终止;否则,将这一元素集合分割成两个或更多个子集合,对每一个子集合分别排序,然后将排好序的子集合归并为一个集合。 3 Z! l" N8 m2 X! o4 g 6 G) Z$ A6 t3 Z9 [' v7 L假设仅将n 个元素的集合分成两个子集合。现在需要确定如何进行子集合的划分。一种可能性就是把前面n- 1个元素放到第一个子集中(称为A),最后一个元素放到第二个子集里(称为B)。按照这种方式对A递归地进行排序。由于B仅含一个元素,所以它已经排序完毕,在A排完序后,只需要用程序2 - 1 0中的函数i n s e r t将A和B合并起来。把这种排序算法与I n s e r t i o n S o r t(见程序2 - 1 5)进行比较,可以发现这种排序算法实际上就是插入排序的递归算法。该算法的复杂性为O (n 2 )。把n 个元素划分成两个子集合的另一种方法是将含有最大值的元素放入B,剩下的放入A中。然后A被递归排序。为了合并排序后的A和B,只需要将B添加到A中即可。假如用函数M a x(见程序1 - 3 1)来找出最大元素,这种排序算法实际上就是S e l e c t i o n S o r t(见程序2 - 7)的递归算法。* I( F6 b3 E: S/ P7 ^" g0 T" U% O1 n) \
/ @; K6 f& i4 K0 `, G n% p假如用冒泡过程(见程序2 - 8)来寻找最大元素并把它移到最右边的位置,这种排序算法就是B u b b l e S o r t(见程序2 - 9)的递归算法。这两种递归排序算法的复杂性均为(n2 )。若一旦发现A已经被排好序就终止对A进行递归分割,则算法的复杂性为O(n2 )(见例2 - 1 6和2 - 1 7)。 ' u3 ^% v! ?- N, U% G7 ^ . W. w: t) O' R" a& t2 H上述分割方案将n 个元素分成两个极不平衡的集合A和B。A有n- 1个元素,而B仅含一个元素。下面来看一看采用平衡分割法会发生什么情况: A集合中含有n/k 个元素,B中包含其余的元素。递归地使用分而治之方法对A和B进行排序。然后采用一个被称之为归并( m e rg e)的过程,将已排好序的A和B合并成一个集合。5 U% ~6 p! f [) t
# a# H7 C/ S9 Q. G) a$ n
例2-5 考虑8个元素,值分别为[ 1 0,4,6,3,8,2,5,7 ]。如果选定k = 2,则[ 1 0 , 4 , 6 , 3 ]和[ 8 , 2 , 5 , 7 ]将被分别独立地排序。结果分别为[ 3 , 4 , 6 , 1 0 ]和[ 2 , 5 , 7 , 8 ]。从两个序列的头部开始归并这两个已排序的序列。元素2比3更小,被移到结果序列;3与5进行比较,3被移入结果序列;4与5比较,4被放入结果序列;5和6比较,.。如果选择k= 4,则序列[ 1 0 , 4 ]和[ 6 , 3 , 8 , 2 , 5 , 7 ]将被排序。排序结果分别为[ 4 , 1 0 ]和[ 2 , 3 , 5 , 6 , 7 , 8 ]。当这两个排好序的序列被归并后,即可得所需要的排序序列。( X0 C! \8 T* q" Z% @! f+ M" {
4 h8 z Z+ z0 q4 ~- K图2 - 6给出了分而治之排序算法的伪代码。算法中子集合的数目为2,A中含有n/k个元素。+ [3 t# s D1 U! B% Y/ T
3 ?. n( n9 H& o+ `, m# M% B( E
template<CLASS T>) S6 }- W9 r: ?" m' B
9 s3 ?& O' [; }+ H
void sort( T E, int n), [) a, [& y: ^
( Z! f0 \1 b& s# v l
{ / /对E中的n 个元素进行排序, k为全局变量 ) E7 I) Z5 z5 O i6 v* s5 ?! b- f# v! ]
if (n >= k) { # b) A6 v; U6 \+ [. h+ d% H9 m
i = n/k; : h) O9 b# t: u# H# K7 P: V: ^* b" c# X4 g: I. d
j = n-i; I! ~! c( n& f" } T' j* s% A! o4 X: _+ \+ c5 R
令A 包含E中的前i 个元素 W% W$ K, L2 ^: F ! H' a4 [ V1 [3 j令B 包含E中余下的j 个元素; k. k! K3 Y( F; b! ]& u; g6 U
2 T3 N! Y% P& b' ?" C
s o r t ( A , i ) ; : ]3 l; v( r+ l: f$ R" E7 y+ T' Q5 X/ \6 b
s o r t ( B , j ) ;- R( e( o7 s3 \
* ^0 \: t* c: r B9 Rm e rge(A,B,E,i,j,); //把A 和B 合并到E 1 ~% S$ Q( B4 Y1 O" l {/ i9 W9 Y6 H7 P
} , n1 y7 H' B& p+ Z ' k7 `9 L$ s! pelse 使用插入排序算法对E 进行排序 * q* p3 Q9 V0 c s: s- y5 w' a+ x# v& X2 B6 o
} - I& c* ~5 y0 y: u; k1 n5 R% v% x, A, ~+ f0 U5 |
图14-6 分而治之排序算法的伪代码 + I# m% H" ^0 N5 W& o 6 h# R" M* f, \/ }0 @8 O3 X1 r+ l9 e7 @2 r$ h
. I. r3 C) W- W6 k# z/ i7 N
从对归并过程的简略描述中,可以明显地看出归并n个元素所需要的时间为O (n)。设t (n)为分而治之排序算法(如图1 4 - 6所示)在最坏情况下所需花费的时间,则有以下递推公式:1 P1 Q' C9 U9 Q' \& j+ p1 ^
. ]' s9 x. \* S/ m/ Z4 A/ J其中c 和d 为常数。当n / k≈n-n / k 时,t (n) 的值最小。因此当k= 2时,也就是说,当两个子集合所包含的元素个数近似相等时, t (n) 最小,即当所划分的子集合大小接近时,分而治之算法通常具有最佳性能。 . r1 e- @/ n2 I) U" g' B! S9 D, w. k. H* F @
可以用迭代方法来计算这一递推方式,结果为t(n)= (nl o gn)。虽然这个结果是在n为2的幂时得到的,但对于所有的n,这一结果也是有效的,因为t(n) 是n 的非递减函数。t(n) =(nl o gn) 给出了归并排序的最好和最坏情况下的复杂性。由于最好和最坏情况下的复杂性是一样的,因此归并排序的平均复杂性为t (n)= (nl o gn)。& m1 P/ c: T3 }1 N- f
) m: Y9 O/ L# O1 k
图2 - 6中k= 2的排序方法被称为归并排序( m e rge sort ),或更精确地说是二路归并排序(two-way merge sort)。下面根据图1 4 - 6中k= 2的情况(归并排序)来编写对n 个元素进行排序的C + +函数。一种最简单的方法就是将元素存储在链表中(即作为类c h a i n的成员(程序3 -8))。在这种情况下,通过移到第n/ 2个节点并打断此链,可将E分成两个大致相等的链表。- v+ Z" I$ o) @
6 ^/ N0 ]* [* P \6 z归并过程应能将两个已排序的链表归并在一起。如果希望把所得到C + +程序与堆排序和插入排序进行性能比较,那么就不能使用链表来实现归并排序,因为后两种排序方法中都没有使用链表。为了能与前面讨论过的排序函数作比较,归并排序函数必须用一个数组a来存储元素集合E,并在a 中返回排序后的元素序列。为此按照下述过程来对图1 4 - 6的伪代码进行细化:当集合E被化分成两个子集合时,可以不必把两个子集合的元素分别复制到A和B中,只需简单地在集合E中保持两个子集合的左右边界即可。接下来对a 中的初始序列进行排序,并将所得到的排序序列归并到一个新数组b中,最后将它们复制到a 中。图1 4 - 6的改进版见图1 4 - 7。 6 E6 R9 M0 i+ `; r1 U& |" @ 0 Q s1 p" u4 E, t2 j" m" v' D G0 Z! F' @: E0 @+ ` 4 s& S5 r5 D. o+ A5 _9 E9 u3 T$ Etemplate<CLASS T> 6 ^6 ] }3 r7 ^0 y @4 b9 Q" e2 v( n6 F A6 B/ |0 _; F' ?
M e rgeSort( T a[], int left, int right)" z7 `3 S% y F3 s9 d; i* L: `
' m& n2 `8 M) X* P7 W3 g{ / /对a [ l e f t : r i g h t ]中的元素进行排序 & i/ \. |' Q5 ~- R# Z% `/ F, n2 s& f* w$ L. }/ X( C8 ?
if (left < right) {//至少两个元素' _, M" o8 w. F, Y! h1 K
6 m' {! z5 N* ^; J; V3 y
int i = (left + right)/2; //中心位置 ' |4 H4 J# h0 h0 r 7 x) t2 ?9 h' O! X2 IM e rgeSort(a, left, i);* g; }( Z; Z" z, g$ }4 |3 h `$ G
9 o8 Z r- ?/ d, E% N, l+ ]" F0 j
M e rgeSort(a, i+1, right); - F5 F: ~! s* R$ g _9 Z( R) {& D$ i0 r, D5 y; P
M e rge(a, b, left, i, right); //从a 合并到b 1 m3 T; R/ f# G! ]9 k* e, ~6 |2 Z & a" o. N# V9 E1 ?. wCopy(b, a, left, right); //结果放回a8 P' V8 b: V8 K
0 K$ f* g$ W+ |' x6 c& N4 a+ }
} 0 c8 [! ?3 A- e) d3 f# `9 y3 q5 @
} / }! K: p! q7 J4 }, m 6 `2 U) M+ s2 h+ ?& j' K图14-7 分而治之排序算法的改进* h$ b" k9 Z; l5 Z b3 S9 v" i$ n2 {8 ]
) |& Y3 ]7 _" ^4 \{// 使用归并排序算法对a[0:n-1] 进行排序0 T- [" |2 I$ e8 a
1 _3 h; r. ~" f
T *b = new T [n];* [, q! o: V5 p, v) ^- J
/ j4 T4 b3 l7 M. |" c
int s = 1; // 段的大小 * ^' ]8 y3 e( T' u6 M2 `4 T& f9 F5 u' g& R/ i/ p d- t1 j
while (s < n) { - x) L4 {# e# H9 h/ s. e5 S. q$ ~. L
MergePass(a, b, s, n); // 从a归并到b9 ^( z. o% m* S$ T6 F4 w, Y
7 g8 r: l% N8 e& |s += s;) y. Q- ?, w( W8 j$ `8 ~
o! `8 A, D3 Y& O
MergePass(b, a, s, n); // 从b 归并到a / d& J; ^' |1 R1 }6 P0 a + ?7 f) d: D0 E" ms += s;' F! G( Z$ ]2 a' b6 U+ W+ f8 ~
" ?( ?# h( `9 E: f) L5 M. E9 Z
}* U G4 s$ u/ d9 C
E- _* V- S# I2 @: p9 C} . u% r, j) i, W! P6 s8 y . g) Z' v6 p. M2 ?为了完成排序代码,首先需要完成函数M e rg e P a s s。函数M e rg e P a s s(见程序1 4 - 4)仅用来确定欲归并子序列的左端和右端,实际的归并工作由函数M e rg e (见程序1 4 - 5 )来完成。函数M e rg e要求针对类型T定义一个操作符< =。如果需要排序的数据类型是用户自定义类型,则必须重载操作符< =。这种设计方法允许我们按元素的任一个域进行排序。重载操作符< =的目的是用来比较需要排序的域。* ^% M4 `0 K- x
# Z# K0 Y" `- n% M( l# O
程序14-4 MergePass函数. j) p0 }& a( r* i( ]- y2 d
0 n0 l* Q; ~& Etemplate<CLASS T> ) [$ i- K, q" s 0 O/ T0 E2 x' B9 pvoid MergePass(T x[], T y[], int s, int n) ' `( A5 P1 U- _ : ~% J) E! G7 n0 A- {4 r4 m4 G2 C{// 归并大小为s的相邻段) [# J9 c% U( z8 o
% ^$ \* N4 x& x5 F8 |
int i = 0; {& `" E$ a/ [& X; F0 z8 J3 ~$ j3 \3 |; u; J% p
while (i <= n - 2 * s) { 0 ~/ H* H v# p! }5 C & b+ P" z9 @4 X7 K+ }! C+ s// 归并两个大小为s的相邻段 5 \& z' v* ~& g7 x) \9 ^: f; ^7 y$ W* [) K
Merge(x, y, i, i+s-1, i+2*s-1); ! B: p1 g! u( m, T) f4 ^; A1 z$ L/ D& K0 g; R: N9 q
i = i + 2 * s;4 B8 K( \* T. j1 H% e; A
" N3 n6 A' m) M
} $ ]7 u6 L1 D& |9 e. g# ^4 p% c. x$ C0 K" b- s
// 剩下不足2个元素* h) V8 Q1 n# e5 [+ i
$ ~. Y: X7 x8 z$ R$ T' A
if (i + s < n) Merge(x, y, i, i+s-1, n-1);7 E& i6 H: E& R
; G; {7 y5 ^. V) d$ s
else for (int j = i; j <= n-1; j++)1 }# t% C- r+ X4 m
0 T4 c$ r7 ?% G; G! ~# y, x8 D// 把最后一段复制到y 8 Q3 w* ^( \7 M' q) W( ]; i! ] 5 T' Z. n k# Q/ E: Ry[j] = x[j]; ; w q. M5 C/ s+ N$ P* ` ' p- r: B2 \5 l" Q* M- Z} # m, Q4 Y( p* F# a 7 Z3 Z5 {4 D% v6 S; p9 b9 A) `程序14-5 Merge函数 / c" o& p( p# x7 `5 G3 z0 a) B: {* Q
template<CLASS T> - K% D4 r( ~# e" Y 7 ]1 N+ ^' u6 o, h' C5 Uvoid Merge(T c[], T d[], int l, int m, int r)5 x% `3 d3 D$ C% P! |7 J" W
; o8 m6 p2 c+ ]$ @& }0 R
{// 把c[l:m]] 和c[m:r] 归并到d [ l : r ] .! M! c) P2 R0 }5 T* b. |
. [7 I3 `% k, m: A
int i = l, // 第一段的游标+ {+ g- L; p9 e Z0 w/ j% W
# F+ ~) {& [+ n4 q, q
j = m+1, // 第二段的游标8 }9 s @, O/ {& s7 m
" f9 e, s" [+ O4 ^, e& S: o/ w5 q
k = l; // 结果的游标 ; X1 ]4 E7 Y3 q# {* W: D( P' ]$ k/ u8 N! o
/ /只要在段中存在i和j,则不断进行归并 2 l G$ p% U" N , K. [- g# F Y% k" Vwhile ((i <= m) && (j <= r))9 X; c* E. k$ O0 `2 J4 o
' v9 p, V/ G. {0 ^4 L# I
if (c <= c[j]) d[k++] = c[i++]; , I8 X+ U5 W5 j + B9 H% j8 X" \) Y3 u- A* m5 g7 oelse d[k++] = c[j++]; : |% B$ O. g& s3 A1 ~+ Z # J$ E& g6 b9 f9 a6 C// 考虑余下的部分 - a) {5 j- f" ~1 O4 w% t8 Z/ D6 O4 b/ y
if (i > m) for (int q = j; q <= r; q++)8 m& L; _) R2 y% s. ]
: E z% b' J [& B6 }8 P
d[k++] = c[q]; 2 W* e" Z* w+ Y( _6 v' ~: _, a3 c2 X. \0 R4 I, e
else for (int q = i; q <= m; q++); K( _% r1 L( b1 E0 B