Prim算法

/**/ /**
 * PRIM(简单版) 最小生成树算法 (Minimum Spanning Tree) 
 * 输入:图g;                 // 有向图或者无向图 
 * 输出:(1)最小生成树长sum; 
 *         (2)最小生成树prev。
 * 结构: 图g用邻接矩阵表示,最短边长dist用数组表示。 
 * 算法:Prim算法  
 * 复杂度:O(|V|^2) 
 
*/
 
#include 
< iostream >
#include 
< vector >
#include 
< list >
#include 
< iterator >
#include 
< algorithm >
#include 
< numeric >
#include 
< functional >
#include 
< climits >
using   namespace  std;

int  n;                     //  n : 顶点个数 
vector < vector < int >   >  g;  //  g : 图(graph)(用邻接矩阵(adjacent matrix)表示) 
vector < bool >  known;         //  known : 各点是否已经选取 
vector < int >  dist;         //  dist : 已选取点集到未选取点的最小边长 
vector < int >  prev;         //  prev : 最小生成树中各点的前一顶点
int  s;                     //  s : 起点(start) 
int  sum;                 //  sum : 最小生成树长 

bool  Prim()                 //  贪心算法(Greedy Algorithm) 
{
    known.assign(n, 
false);
    dist.assign(n, INT_MAX);
    prev.resize(n);            
// 初始化known、dist、prev。 
    dist[s] = 0;            // 初始化起点到自身的路径长为0。
    int i; 
    
for (i = 0; i < n; ++i)
    
{
        
int min = INT_MAX, v;
        
for (int i = 0; i < n; ++i)
            
if (!known[i] && min > dist[i])
                min 
= dist[i], v = i;    // 寻找未知的最短路径长的顶点v, 
        if (min == INT_MAX) break;        // 如果找不到,退出; 
        known[v] = true;                // 如果找到,将顶点v设为已知,
        sum += dist[v];                 // 调整最小生成树长 
        for (int w = 0; w < n; ++w)        // 遍历所有v指向的顶点w, 
            if (!known[w] && g[v][w] < INT_MAX && dist[w] > g[v][w])
                dist[w] 
= g[v][w], prev[w] = v;    // 调整顶点w的最短路径长dist和最短路径的前一顶点 prev。 
    }

    
return i == n; // 如果选取顶点个数为n,成功。 
}


int  main()
{
    n 
= 7;
    g.assign(n, vector
<int>(n, INT_MAX));
    g[
0][1= g[1][0= 2; g[0][2= g[2][0= 4;   g[0][3= g[3][0= 1;
    g[
1][3= g[3][1= 3; g[1][4= g[4][1= 10;
    g[
2][3= g[3][2= 2; g[2][5= g[5][2= 5;
    g[
3][4= g[4][3= 7; g[3][5= g[5][3= 8; g[3][6= g[6][3= 4;
    g[
4][6= g[6][4= 6;
    g[
5][6= g[6][5= 1
    
    s 
= 0;        // 起点任选 
    sum = 0;
    
if (Prim())
    
{
        cout 
<< sum << endl;
        
for (int i = 1; i < n; ++i)
            
if(i != s) cout << prev[i] << "->" << i << endl;
    }

    
else
    
{
        cout 
<< "Some vertex cann't be reached." << endl; 
    }

    
    system(
"pause");
    
return 0;
}

你可能感兴趣的:(Prim算法)