23最小生成树之Kruskal算法

图的最优化问题:最小生成树、最短路径

典型的图应用问题

无向连通加权图的最小生成树

有向/无向加权图的最短路径

四个经典算法

Kruskal算法、Prim算法---------------最小生成树

Dijkstra算法、Floyd算法-------------最短路径

 

最小生成树的概念:

G=(V,E):无向连通加权图

C(e)或C(v,w): 边e=(v,w)的耗费(Cost)

若S=(V,T)是G的一棵生成树(T是树边集),那么, S的边长之和称作生成树S的耗费C(S)

耗费C(S)达到最小值的生成树S,称为G的最小耗费生成树,也称最小生成树(MinimumSpanningTree)

即C(S)≤C(S') (S'是图G的任一生成树)

注意:最小生成树可能不唯一

 

Kruskal算法

算法思想: 按长度从小到大的依次把边加进生成树,若添加某边后形成了回路,就舍弃这条边

反复如此,直到选出n-1条边,便得到最小生成树.。

 

 

通俗来讲,该算法很简单。

步骤:

1.在空白处画出与原图一模一样的结点位置。(只画结点,权不需要,结点的位置要一模一样)。

2.按权画边。从权值最小的开始,如图,从4开始,草图就连A-B,到权值5,就连B-F,到权值6,此时就会发现,如果连接A-F,那么A-B-F就会构成一个完整的回路(从任意一点出发可以回到原点。)。因此,就要舍去权值6,即边A-F不可连接。。。。。依次按权大小画其他边。直到最后一个结点。

 

 

Kruskal算法伪程序形式

void  Kruskal(***) //***表示函数要求的参数

  {  int et=0;           //et用于记录选中的边数

   置树边集T为空;

   while(et<n-1)               //n是图中的顶点数

    { 从G中选出当前最短边(v,w);

      if(添此边于T中,不使生成树产生回路)

        { 把(v,w)加进T;et++;}

      else  舍弃(v,w);

       }

  }

 

实现方法分析:

 

 

 

子树合并法描述和示例

 

 

 

描述:

 

 

 

Kruskal算法的子树合并法描述形式:

方法一:

void  Kruskal(***) //***表示函数要求的参数

 {  int et=0;

   每个顶点自成一个集合,并指定集合名;

   while(et<n-1)

     { 从G中选出当前最短边(v,w);

       找到v所在的集合名i,和w所在的集合名j;

         if(i!=j)

         {  把(v,w)加进T;  et++;

           将集合i与集合j合并成一个集合k;     }

      }

  }

方法二:

void  Kruskal(***) //***表示函数要求的参数

  {  int et=0;          

  每个顶点一个集合,并指定集合名 ;

   while(et<n-1)

    { 从G中选出当前最短边(v,w);

      if(v和w不在同一子树中)

        { 把(v,w)加进T;et++;

        将v所在的子树与w所在的子树合并成一棵子树;

           }

      }

  }

 

结论:无论是按权画边法,还是子树合并法都是可以找出最小生成树,但是个人认为按权画边法比较容易理解。

 

posted @ 2018-03-05 16:08  gd_沐辰  阅读(218)  评论(0编辑  收藏  举报