<>在例1 - 2及1 - 3中已考察过这个问题。因为具有n 个顶点的无向网络G的每个生成树刚好具有n-1条边,所以问题是用某种方法选择n-1条边使它们形成G的最小生成树。至少可以采用三种不同的贪婪策略来选择这n-1条边。这三种求解最小生成树的贪婪算法策略是: K r u s k a l算法,P r i m算法和S o l l i n算法。</P>8 L7 Y; q& i$ {( q
<>1. Kruskal算法</P> " H! v8 P$ w" t( w0 l" k: f& d* Y<>(1) 算法思想</P> 7 Q) C* G* j3 {<>K r u s k a l算法每次选择n- 1条边,所使用的贪婪准则是:从剩下的边中选择一条不会产生环路的具有最小耗费的边加入已选择的边的集合中。注意到所选取的边若产生环路则不可能形成一棵生成树。K r u s k a l算法分e 步,其中e 是网络中边的数目。按耗费递增的顺序来考虑这e 条边,每次考虑一条边。当考虑某条边时,若将其加入到已选边的集合中会出现环路,则将其抛弃,否则,将它选入。</P>( \2 ~) {- Y( E
<>考察图1-12a 中的网络。初始时没有任何边被选择。图13-12b 显示了各节点的当前状态。边( 1 , 6)是最先选入的边,它被加入到欲构建的生成树中,得到图1 3 - 1 2 c。下一步选择边( 3,4)并将其加入树中(如图1 3 - 1 2 d所示)。然后考虑边( 2,7 ),将它加入树中并不会产生环路,于是便得到图1 3 - 1 2 e。下一步考虑边( 2,3)并将其加入树中(如图1 3 - 1 2 f所示)。在其余还未考虑的边中,(7,4)具有最小耗费,因此先考虑它,将它加入正在创建的树中会产生环路,所以将其丢弃。此后将边( 5,4)加入树中,得到的树如图13-12g 所示。下一步考虑边( 7,5),由于会产生环路,将其丢弃。最后考虑边( 6,5)并将其加入树中,产生了一棵生成树,其耗费为9 9。图1 - 1 3给出了K r u s k a l算法的伪代码。</P> / j( a$ g7 u8 C6 \: U8 ]* @$ p& C% x) L8 ~6 j
' R# B$ @% S6 v* S& S' h( ?" T
<>/ /在一个具有n 个顶点的网络中找到一棵最小生成树</P> ! t3 V2 b: I; l<>令T为所选边的集合,初始化T=</P> # y, [+ N% L7 f* l+ y5 E- X* @<>令E 为网络中边的集合</P>5 _0 p& d3 D: K/ u* C7 ~0 ?% F+ L
<>w h i l e(E≠ )&&(| T |≠n- 1 ) {</P>% v; `# K7 k' |0 r; W2 y( y) T
<>令(u,v)为E中代价最小的边 E=E- { (u,v) } / /从E中删除边</P> 6 \7 P. P) L6 w2 Y' U- F7 x) |<>i f( (u,v)加入T中不会产生环路)将( u,v)加入T</P>6 h& c' U% t* Z8 K
<>}</P>$ \9 X, A5 x H: N
<>i f(| T | = =n-1) T是最小耗费生成树</P>- Q) l/ x: t; U$ n- }
<>e l s e 网络不是互连的,不能找到生成树</P>1 K' B' y) N8 S9 o7 f: ?% T
<>图13-13 Kruskao算法的伪代码</P> # Z# g% W& f) M, v1 q<p>3 @4 ? y M P; v: R
<>(2) 正确性证明</P> 8 t6 J5 g; E9 c" V5 e7 N<>利用前述装载问题所用的转化技术可以证明图1 3 - 1 3的贪婪算法总能建立一棵最小耗费生成树。需要证明以下两点: 1) 只要存在生成树,K r u s k a l算法总能产生一棵生成树; 2) 产生的生成树具有最小耗费。令G为任意加权无向图(即G是一个无向网络)。从1 2 . 11 . 3节可知当且仅当一个无向图连通时它有生成树。而且在Kruskal 算法中被拒绝(丢弃)的边是那些会产生环路的边。删除连通图环路中的一条边所形成的图仍是连通图,因此如果G在开始时是连通的,则T与E中的边总能形成一个连通图。也就是若G开始时是连通的,算法不会终止于E= 和| T |< n- 1。</P> % S# o- {8 v4 `. o' H. y) d; @<>现在来证明所建立的生成树T具有最小耗费。由于G具有有限棵生成树,所以它至少具有一棵最小生成树。令U为这样的一棵最小生成树, T与U都刚好有n- 1条边。如果T=U, 则T就具有最小耗费,那么不必再证明下去。因此假设T≠U,令k(k >0) 为在T中而不在U中的边的个数,当然k 也是在U中而不在T中的边的数目。 通过把U变换为T来证明U与T具有相同的耗费,这种转化可在k 步内完成。每一步使在T而不在U中的边的数目刚好减1。而且U的耗费不会因为转化而改变。经过k 步的转化得到的U将与原来的U具有相同的耗费,且转化后U中的边就是T中的边。由此可知, T具有最小耗费。每步转化包括从T中移一条边e 到U中,并从U中移出一条边f。边e 与f 的选取按如下方式进行:</P> 6 |0 Q8 H' Z& J<>1) 令e 是在T中而不在U中的具有最小耗费的边。由于k >0,这条边肯定存在。</P> b, f; c6 R6 Y) M5 _<>2) 当把e 加入U时,则会形成唯一的一条环路。令f 为这条环路上不在T中的任意一条边。</P> " P) l! k H! |* K, F<>由于T中不含环路,因此所形成的环路中至少有一条边不在T中。</P> 5 i+ L$ l/ c0 r2 Y7 m" P' J# b' x<>从e 与f 的选择方法中可以看出, V=U+ {e} -{ f } 是一棵生成树,且T中恰有k- 1条边不在V中出现。现在来证明V的耗费与U的相同。显然,V的耗费等于U的耗费加上边e 的耗费再减去边f 的耗费。若e 的耗费比f 的小,则生成树V的耗费比U的耗费小,这是不可能的。如果e 的耗费高于f,在K r u s k a l算法中f 会在e 之前被考虑。由于f 不在T中,Kruskal 算法在考虑f 能否加入T时已将f 丢弃,因此f 和T中耗费小于或等于f 的边共同形成环路。通过选择e,所有这些边均在U中,因此U肯定含有环路,但是实际上这不可能,因为U是一棵生成树。e 的代价高于f 的假设将会导致矛盾。剩下的唯一的可能是e 与f 具有相同的耗费,由此可知V与U的耗费相同。</P>8 |1 Y# s( _7 K; Z2 \2 Z6 J/ G
<>(3) 数据结构的选择及复杂性分析</P> # X( t) L, ]3 t q3 M: R) U<>为了按耗费非递减的顺序选择边,可以建立最小堆并根据需要从堆中一条一条地取出各边。当图中有e 条边时,需花(e) 的时间初始化堆及O ( l o ge) 的时间来选取每一条边。边的集合T与G中的顶点一起定义了一个由至多n 个连通子图构成的图。用顶点集合来描述每个子图,这些顶点集合没有公共顶点。为了确定边( u,v)是否会产生环路,仅需检查u,v 是否在同一个顶点集中(即处于同一子图)。如果是,则会形成一个环路;如果不是,则不会产生环路。因此对于顶点集使用两个F i n d操作就足够了。当一条边包含在T中时,2个子图将被合并成一个子图,即对两个集合执行U n i o n操作。集合的F i n d和U n i o n操作可利用8 . 1 0 . 2节的树(以及加权规则和路径压缩)来高效地执行。F i n d操作的次数最多为2e,Un i o n操作的次数最多为n- 1(若网络是连通的,则刚好是n- 1次)。加上树的初始化时间,算法中这部分的复杂性只比O (n+e) 稍大一点。</P>8 D/ @, X5 n# \ N8 e
<>对集合T所执行的唯一操作是向T中添加一条新边。T可用数组t 来实现。添加操作在数组</P>+ E! G6 c7 w" D6 N( t
<>的一端进行,因为最多可在T中加入n- 1条边,因此对T的操作总时间为O (n)。</P>/ r" R2 O! q: v, [( S0 R- s6 }
<>总结上述各个部分的执行时间,可得图1 3 - 1 3算法的渐进复杂性为O (n+el o ge)。</P>0 V& H6 w: h8 y& h: I9 ?) |) g
<>(4) 实现</P>7 Q8 W1 I4 T3 C8 ?
<>利用上述数据结构,图1 - 1 3可用C + +代码来实现。首先定义E d g e N o d e类(见程序1 3 - 6 ),它是最小堆的元素及生成树数组t 的数据类型。</P>0 E0 ~9 q9 _3 T1 D% L8 _3 K
<>程序13-6 Kruskal算法所需要的数据类型</P>0 L6 l* d: Q" c7 [/ n
<P>template <CLASS T></P>8 l+ C, ~' j. l) u7 s7 \
<P>class EdgeNode {</P>$ q l$ F+ O: P: u
<P>p u b l i c :</P> 0 U5 E/ _6 z5 Q& U<P>operator T() const {return weight;}</P>2 i& x/ M9 a8 ^5 o+ g
<P>p r i v a t e :</P>% l6 j8 a/ G7 E! m
<P>T weight;//边的高度</P>9 ?3 n9 n$ @0 F9 H$ `) E& p
<P>int u, v;//边的端点</P> / P. N# ~" V0 o$ ]5 F+ ?- Q ?* r<P>} ;</P>4 z1 P, _/ O Z" Z9 R
<P>为了更简单地使用8 . 1 0 . 2节的查找和合并策略,定义了U n i o n F i n d类,它的构造函数是程序8 - 1 6的初始化函数,U n i o n是程序8 - 1 6的加权合并函数,F i n d是程序8 - 1 7的路径压缩搜索函数。</P>& c1 o7 c4 j- J; U0 b
<P>为了编写与网络描述无关的代码,还定义了一个新的类U N e t Wo r k,它包含了应用于无向网络的所有函数。这个类与U n d i r e c t e d类的差别在于U n d i r e c t e d类中的函数不要求加权边,而U N e t Wo r k要求边必须带有权值。U N e t Wo r k中的成员需要利用N e t w o r k类中定义的诸如B e g i n和N e x t Ve r t e x的遍历函数。不过,新的遍历函数不仅需要返回下一个邻接的顶点,而且要返回到达这个顶点的边的权值。这些遍历函数以及有向和无向加权网络的其他函数一起构成了W N e t w o r k类(见程序1 3 - 7)。</P> & ~* f {# m$ [% r; m) q<P>程序13-7 WNetwork类</P> 0 l# U4 ~+ r( a$ A, n% s+ P<P>template<CLASS T></P>* D0 n B% i. o" y* ~' V
<P>class WNetwork : virtual public Network</P> v+ S: v0 y; m% ^- A. C
<P>{</P> % ?6 [8 K5 s- o# u4 g<P>public :</P>: z( i$ s" Z: U1 Z- J
<P>virtual void First(int i, int& j, T& c)=0;</P> 8 z. y- ]3 y* @<P>virtual void Next(int i, int& j, T& c)=0;</P> [) I; C# j# r% v8 y5 k<P>} ;</P> T, L- S( q/ C<P>象B e g i n和N e x t Ve r t e x一样,可在A d j a c e n c y W D i g r a p h及L i n k e d W D i g r a p h类中加入函数F i r s t与N e x t。现在A d j a c e n c y W D i g r a p h及L i n k e d W D i g r a p h类都需要从W N e t Wo r k中派生而来。由于A d j a c e n c y W G r a p h类和L i n k e d W G r a p h类需要访问U N e t w o r k的成员,所以这两个类还必须从U N e t Wo r k中派生而来。U N e t Wo r k : : K r u s k a l的代码见程序1 3 - 8,它要求将Edges() 定义为N e t Work 类的虚成员,并且把U N e t Wo r k定义为E d g e N o d e的友元)。如果没有生成树,函数返回f a l s e,否则返回t r u e。注意当返回true 时,在数组t 中返回最小耗费生成树。</P> ) s# r, l1 D$ S<P>程序13-8 Kr u s k a l算法的C + +代码</P> 6 X' X) p2 i6 l<P>template<CLASS T></P>% D' R6 D( C* s# O/ [
<P>bool UNetwork<T>::Kruskal(EdgeNode<T> t[])</P> 5 r$ T' c e5 K" o/ W) v$ x; o5 U' ^<P>{// 使用K r u s k a l算法寻找最小耗费生成树</P> . ^7 H& b4 q2 Q8 l) Q C<P>// 如果不连通则返回false</P> 7 \5 C, H1 X! @' l* w4 j: n<P>// 如果连通,则在t [ 0 : n - 2 ]中返回最小生成树</P> # x* Q0 t7 }% {: W( F" x1 J% Q<P>int n = Ve r t i c e s ( ) ;</P>, q w8 ^3 A& c5 c4 o5 e3 o8 M
<P>int e = Edges();</P>% }6 a4 `% p/ T1 n; P
<P>/ /设置网络边的数组</P> % B" s' n" k- m: j<P>InitializePos(); // 图遍历器</P>0 b# O/ r: f; o- t& N
<P>EdgeNode<T> *E = new EdgeNode<T> [e+1];</P> 7 B+ b1 t5 c. T: F! l5 T<P>int k = 0; // E的游标</P>+ s( T. j. P+ A& u4 e) J* i
<P>for (int i = 1; i <= n; i++) { // 使所有边附属于i</P> 2 N, }+ ]" V& Z<P>int j;</P> & Z7 m7 @4 n: `3 ^; W) a<P>T c;</P>/ U7 L' |* a& C
<P>First(i, j, c);</P>7 r$ c: ?8 [- r- A g7 r9 ?$ w
<P>while (j) { // j 邻接自i</P>0 l' m' I) V- B* [4 r6 ~0 N
<P>if (i < j) {// 添加到达E的边</P> ( G0 l7 t5 E5 U8 n- C8 F) u<P>E[++k].weight = c;</P>9 N( R. X1 s1 ^* g8 W& v
<P>E[k].u = i;</P> ' g6 m9 M. s" m" t8 Q+ }8 A<P>E[k].v = j;}</P> ! e2 a0 `2 ^$ B* T" b<P>Next(i, j, c);</P> : I" _% N. Y, @ \: w1 ], h<P>}</P># e& R: G3 t) \3 C' o
<P>}</P> 9 U) o% \' {, K<P>// 把边放入最小堆</P>! j. {: O' Z: x
<P>MinHeap<EDGENODE<T> > H(1);</P> 2 j: a; w$ } T: [0 Q, r6 O<P>H.Initialize(E, e, e);</P>( d6 ~: [5 }% N4 ~; t
<P>UnionFind U(n); // 合并/搜索结构</P>+ l) k l: v8 h# ^: Q
<P>// 根据耗费的次序来抽取边</P> * `: v% U0 ^1 W$ j5 J Y<P>k = 0; // 此时作为t的游标</P> # O2 Y: {4 z( h$ g<P>while (e && k < n - 1) {</P> 6 F @- O5 p3 j& D# o: k: S* w& }<P>// 生成树未完成,尚有剩余边</P> 5 T+ H) m: Q7 ^& H3 E<P>EdgeNode<T> x;</P> - W M+ \, x/ j# Z8 u9 n<P>H.DeleteMin(x); // 最小耗费边</P>* W6 u$ R. N0 _* ?' x4 L2 q
<P>e - - ;</P> * a* E0 P* L1 R# j<P>int a = U.Find(x.u);</P>4 s+ D o# x# ~
<P>int b = U.Find(x.v);</P> " g. P/ E' m/ Z% K<P>if (a != b) {// 选择边</P> 4 t- l2 v1 g7 L<P>t[k++] = x;</P>5 V s7 p5 n% B4 }
<P>U . U n i o n ( a , b ) ; }</P> " {- s) H0 \6 x; L<P>}</P>( A( |. y/ [. k7 t
<P>D e a c t i v a t e P o s ( ) ;</P>& S1 I9 e: k& l+ M- z0 W5 n
<P>H . D e a c t i v a t e ( ) ;</P> 6 B; ?: G4 o0 u5 s<P>return (k == n - 1);</P> # g! p* u; B% u! G<P>}</P> % ~# x# g2 d. q$ @: u<P>2. Prim算法</P> s' r! ^1 A4 ]; x: E5 |<P>与Kr u s k a l算法类似,P r i m算法通过每次选择多条边来创建最小生成树。选择下一条边的贪婪准则是:从剩余的边中,选择一条耗费最小的边,并且它的加入应使所有入选的边仍是一棵树。最终,在所有步骤中选择的边形成一棵树。相反,在Kruskal 算法中所有入选的边集合最终形成一个森林。</P>. H. f6 M" K ~% g5 d/ l8 Y
<P>P r i m算法从具有一个单一顶点的树T开始,这个顶点可以是原图中任意一个顶点。然后往T中加入一条代价最小的边( u , v)使Tè{ (u , v) }仍是一棵树,这种加边的步骤反复循环直到T中包含n- 1条边。注意对于边( u , v),u、v 中正好有一个顶点位于T中。P r i m算法的伪代码如图1 -1 4所示。在伪代码中也包含了所输入的图不是连通图的可能,在这种情况下没有生成树。图1 - 1 5显示了对图1-12a 使用P r i m算法的过程。把图1 - 1 4的伪代码细化为C + +程序及其正确性的证明留作练习(练习3 1)。</P> : \: k7 J: l& B5 w% q) B<p>2 _' I* J! C* h: I
<P>/ /假设网络中至少具有一个顶点</P>. E2 a3 c3 ^, o6 Q) ?( M7 O
<P>设T为所选择的边的集合,初始化T=</P>0 w: S9 }) ?2 A& }4 o& a7 t
<P>设T V为已在树中的顶点的集合,置T V= { 1 }</P>5 ~; `. g* S9 f" _
<P>令E为网络中边的集合</P> 0 W% Q. o0 @8 d% M( h<P>w h i l e(E< > ) & & (| T | < > n-1) {</P>3 Q' D& |6 v, u) V7 O
<P>令(u , v)为最小代价边,其中u T V, v T V</P>9 K' b: ?: @5 i. G( G2 T1 o$ S; R
<P>i f(没有这种边) b re a k</P> & j8 [8 q8 z# M0 S( V<P>E=E- { (u,v) } / /从E中删除此边</P> * t# Z4 W7 O! v& W. v$ S) A<P>在T中加入边( u , v)</P> $ U' d. k; B# R. I! E% W<P>}</P> 0 E! P# Z. K8 X. x% x# S<P>if (| T | = =n- 1 ) T是一棵最小生成树</P> % T, L+ h9 e, _+ ]# w$ K<P>else 网络是不连通的,没有最小生成树</P> 7 M' Z& a1 a$ K+ Q<P>图13-14 Prim最小生成树算法</P>. v; @! @ Y0 J; ?7 R v
<p>4 V4 l' ? ^( v7 d+ s
<P>如果根据每个不在T V中的顶点v 选择一个顶点n e ar (v),使得n e ar (v) ? TV 且c o st (v, n e ar (v) )的值是所有这样的n e ar (v) 节点中最小的,则实现P r i m算法的时间复杂性为O (n2 )。下一条添加到T中的边是这样的边:其cost (v, near (v)) 最小,且v T V。</P>& l8 l) }) ^$ p1 D" E4 U# s
<P>3. Sollin算法</P> . ^& b3 J0 J. M+ l$ [0 u<P>S o l l i n算法每步选择若干条边。在每步开始时,所选择的边及图中的n个顶点形成一个生成树的森林。在每一步中为森林中的每棵树选择一条边,这条边刚好有一个顶点在树中且边的代价最小。将所选择的边加入要创建的生成树中。注意一个森林中的两棵树可选择同一条边,因此必须多次复制同一条边。当有多条边具有相同的耗费时,两棵树可选择与它们相连的不同的边,在这种情况下,必须丢弃其中的一条边。开始时,所选择的边的集合为空。若某一步结束时仅剩下一棵树或没有剩余的边可供选择时算法终止。</P>1 y- ^7 k0 X& c4 e
<P>图1 - 6给出了初始状态为图1-12a 时,使用S o l l i n算法的步骤。初始入选边数为0时的情形如图13-12a 时,森林中的每棵树均是单个顶点。顶点1,2,.,7所选择的边分别是(1.6), (2,7),(3,4), (4,3), (5,4), (6,1), (7,2),其中不同的边是( 1 , 6 ),( 2 , 7 ),(3,4) 和( 5 , 4 ),将这些边加入入选边的集合后所得到的结果如图1 3 - 1 6 a所示。下一步具有顶点集{ 1 , 6 }的树选择边( 6 , 5 ),剩下的两棵树选择边( 2 , 3 ),加入这两条边后已形成一棵生成树,构建好的生成树见图1 3 - 6 b。S o l l i n算法的C + +程序实现及其正确性证明留作练习(练习3 2 )。</P>