zoukankan      html  css  js  c++  java
  • 最小生成树MST

    一、最小生成树MST

    1.Kruskal算法

    按照边的权值由小到大排序查看一遍,如果该边的两顶点不在同一连通分量里,则加入最小生成树中,可用并查集高效的判断两点是否在同一连通分量中;

    例:hdu1863畅通工程

    Problem Description
    省政府“畅通工程”的目标是使全省任何两个村庄间都可以实现公路交通(但不一定有直接的公路相连,只要能间接通过公路可达即可)。经过调查评估,得到的统计表中列出了有可能建设公路的若干条道路的成本。现请你编写程序,计算出全省畅通需要的最低成本。
     
    Input
    测试输入包含若干测试用例。每个测试用例的第1行给出评估的道路条数 N、村庄数目M ( < 100 );随后的 N 
    行对应村庄间道路的成本,每行给出一对正整数,分别是两个村庄的编号,以及此两村庄间道路的成本(也是正整数)。为简单起见,村庄从1到M编号。当N为0时,全部输入结束,相应的结果不要输出。
     
    Output
    对每个测试用例,在1行里输出全省畅通需要的最低成本。若统计数据不足以保证畅通,则输出“?”。
     
    Sample Input
    3 3 1 2 1 1 3 2 2 3 4 1 3 2 3 2 0 100
     
    Sample Output
    3 ?
     
    #include <iostream>
    #include <algorithm>
    //kruskal将边排序,再贪心从最小边开始选取,用并查集维护MST的连通性;
    using namespace std;
    typedef long long ll;
    const int maxn=102;
    int n,m;
    struct edge{
        int from,to;
        ll cost;
    };
    edge e[maxn*maxn];
    bool cmp(edge x,edge y)
    {
        return x.cost<y.cost;
    }
    int par[maxn];
    int init()
    {
        for(int i=1;i<=m;i++)
            par[i]=i;
    }
    int fin(int x)
    {
        return par[x]==x?x:par[x]=fin(par[x]);
    }
    bool sam(int x,int y)
    {
        return fin(x)==fin(y);
    }
    void unit(int x,int y)
    {
        x=fin(x),y=fin(y);
        if(x==y) return;
         par[y]=x;
    }
    ll kruskal()
    {
        sort(e+1,e+n+1,cmp);
        init();
        ll res=0;
        for(int i=1;i<=n;i++)
        {
            if(sam(e[i].from,e[i].to))continue; //两顶点已连通;
            unit(e[i].from,e[i].to);
            res+=e[i].cost;
        }
        return res;
    }
    int main()
    {
        while(cin >> n >>m,n)
        {
            ll ans=0;
            for(int i=1;i<=n;i++)
                cin >> e[i].from >> e[i].to >> e[i].cost;
            ans=kruskal();
            for(int i=1;i<=m;i++)
                if(!sam(i,1)){
                    ans=-1;
                }
            if(ans==-1)
                cout << "?" << endl;
            else
            cout << ans << endl;
        }
        return 0;
    }

    例:POJ2421 Constructing Roads

    Description

    There are N villages, which are numbered from 1 to N, and you should build some roads such that every two villages can connect to each other. We say two village A and B are connected, if and only if there is a road between A and B, or there exists a village C such that there is a road between A and C, and C and B are connected. 

    We know that there are already some roads between some villages and your job is the build some roads such that all the villages are connect and the length of all the roads built is minimum.

    Input

    The first line is an integer N (3 <= N <= 100), which is the number of villages. Then come N lines, the i-th of which contains N integers, and the j-th of these N integers is the distance (the distance should be an integer within [1, 1000]) between village i and village j. 

    Then there is an integer Q (0 <= Q <= N * (N + 1) / 2). Then come Q lines, each line contains two integers a and b (1 <= a < b <= N), which means the road between village a and village b has been built.

    Output

    You should output a line contains an integer, which is the length of all the roads to be built such that all the villages are connected, and this value is minimum.

    Sample Input

    3
    0 990 692
    990 0 179
    692 179 0
    1
    1 2
    

    Sample Output

    179
    #include <iostream>
    #include <algorithm>
    #include <vector>
    #include <queue>
    using namespace std;
    typedef long long ll;
    const int maxn=102;
    int n,par[maxn];
    struct edge{
        int from,to;
        ll cost;
        bool operator< (const edge& x)const{
                return x.cost<cost;
        }
    };
    priority_queue<edge> que;
    void init()
    {
        for(int i=1;i<=n;i++)
            par[i]=i;
    }
    int fin(int x){
        return par[x]==x?x:par[x]=fin(par[x]);
    }
    bool sam(int x,int y){
        return fin(x)==fin(y);
    }
    void unit(int x,int y)
    {
        x=fin(x),y=fin(y);
        if(x==y)return;
        par[y]=x;
    }
    ll kruskal()
    {
        ll res=0;
        while(!que.empty())
        {
            edge e=que.top();
            que.pop();
            if(!sam(e.from,e.to))
            {
                res+=e.cost;
                unit(e.from,e.to);
            }
        }
        return res;
    }
    int main()
    {
        int m;
        cin >> n;
        edge e;
        for(int i=1;i<=n;i++)
        {
            for(int j=1;j<=n;j++)
            {
                cin >> e.cost;
                e.from=i,e.to=j;
                if(i!=j)
                    que.push(e);
            }
        }
        init();
        cin >> m;
        for(int i=0;i<m;i++)
        {
            int a,b;
            cin >> a >> b;
            unit(a,b);
        }
        ll ans=kruskal();
        cout << ans << endl;
        return 0;
    }

    2、Prim算法

    首先,假设有一棵只包含一个顶点v的树T,然后贪心的选取T和其他顶点相邻的最小权值的边,并把它加入到T中。不断进行这个操作,就可以得到一棵最小生成树。

     

    例:hdu1863畅通工程

    #include <iostream>
    #include <algorithm>
    #include <queue>
    #include <vector>
    #include <cstring>
    using namespace std;
    typedef long long ll;
    const int maxn=102;
    struct edge{
        int to;
        ll cost;
        edge(int x,ll y):to(x),cost(y){}
        edge(){}
        bool operator< (const edge &x)const{
            return x.cost<cost;
        }
    };
    
    priority_queue<edge> que;   //选取与T相邻代价最小的边;
    vector<edge> G[maxn];       //对每个点存与之相邻的所有边的信息;(邻接表)
    int n,m;
    bool vis[maxn];             //标记已连接的点;
    
    ll prim()
    {
        ll res=0;
        vis[1]=1;               //选取一个初的顶点为树T;
        for(int i=0;i<G[1].size();i++)
            que.push(G[1][i]);  //将与1相邻的所有边放入队列;
        while(que.size())
        {
            edge tmp=que.top();
            que.pop();
            if(vis[tmp.to])continue;
            vis[tmp.to]=1;
            res+=tmp.cost;
            for(int i=0;i<G[tmp.to].size();i++)
                que.push(G[tmp.to][i]);
        }
        return res;
    }
    int main()
    {
        while(cin >> n >> m,n)
        {
            for(int i=0;i<=m;i++)
                G[i].clear();
            while(que.size())que.pop();
            memset(vis,0,sizeof(vis));  //初始化清空
            for(int i=1;i<=n;i++)       //存取所有边的信息;
            {
                int u,v;
                ll cost;
                cin >> u >> v >> cost;
                G[u].push_back(edge(v,cost));
                G[v].push_back(edge(u,cost));
            }
            ll ans=prim();
            for(int i=1;i<=m;i++)       //判断是否连通
                if(!vis[i])
                    ans=-1;
            if(ans==-1)
                cout << "?" << endl;
            else
                cout << ans << endl;
        }
        return 0;
    }

    例:POJ1258 Agri-Net

    Description

    Farmer John has been elected mayor of his town! One of his campaign promises was to bring internet connectivity to all farms in the area. He needs your help, of course. 
    Farmer John ordered a high speed connection for his farm and is going to share his connectivity with the other farmers. To minimize cost, he wants to lay the minimum amount of optical fiber to connect his farm to all the other farms. 
    Given a list of how much fiber it takes to connect each pair of farms, you must find the minimum amount of fiber needed to connect them all together. Each farm must connect to some other farm such that a packet can flow from any one farm to any other farm. 
    The distance between any two farms will not exceed 100,000. 

    Input

    The input includes several cases. For each case, the first line contains the number of farms, N (3 <= N <= 100). The following lines contain the N x N conectivity matrix, where each element shows the distance from on farm to another. Logically, they are N lines of N space-separated integers. Physically, they are limited in length to 80 characters, so some lines continue onto others. Of course, the diagonal will be 0, since the distance from farm i to itself is not interesting for this problem.

    Output

    For each case, output a single integer length that is the sum of the minimum length of fiber required to connect the entire set of farms.

    Sample Input

    4
    0 4 9 21
    4 0 8 17
    9 8 0 16
    21 17 16 0
    

    Sample Output

    28
    #include <iostream>
    #include <queue>
    #include <vector>
    #include <cstring>
    using namespace std;
    typedef long long ll;
    const int maxn=102;
    struct edge{
        int to;
        ll cost;
        edge(int x,ll y):to(x),cost(y){}
        edge(){}
        bool operator< (const edge& xx) const{
            return xx.cost<cost;
        }
    };
    priority_queue<edge> que;
    ll G[maxn][maxn]; //邻接矩阵;
    int n;
    bool vis[maxn];
    ll prim()
    {
        ll res=0;
        vis[1]=1;
        for(int i=2;i<=n;i++)   //该点相邻的点均入队列;
            que.push(edge(i,G[1][i]));
        while(!que.empty())
        {
            edge e=que.top();
            que.pop();
            if(vis[e.to])continue;
            res+=e.cost;
            vis[e.to]=1;
            for(int i=1;i<=n;i++)//新加入的点的相邻边入队列;
                if(i!=e.to)
                    que.push(edge(i,G[e.to][i]));
        }
        return res;
    }
    int main()
    {
        while(cin >> n)
        {
            memset(vis,0,sizeof(vis));
            while(!que.empty()) que.pop();
            for(int i=1;i<=n;i++)
                for(int j=1;j<=n;j++)
                    cin >> G[i][j];
            ll ans=prim();
            cout << ans << endl;
        }
        return 0;
    }
  • 相关阅读:
    根据excel的文件的路径提取其中表的数据到DataSet中 .
    查询SQL Server数据库中的用户表的数量的问题
    C#获取键盘和鼠标操作的时间的类
    使用消息来处理多线程程序中的一些问题
    多线程程序写日志时遇到加锁的问题
    android之横向滚动图
    android listview的一些设置
    android之异步任务
    android之隐式intent调用
    android之广播
  • 原文地址:https://www.cnblogs.com/Cloud-king/p/8899641.html
Copyright © 2011-2022 走看看