zoukankan      html  css  js  c++  java
  • 最小生成树(Kruskal+Prim)--模板

    最小生成树-----在连通网的所有生成树中,所有边的代价和最小的生成树,称为最小生成树。

    应用场景

    1、假设以下情景,有一块木板,板上钉上了一些钉子,这些钉子可以由一些细绳连接起来。假设每个钉子可以通过一根或者多根细绳连接起来,那么一定存在这样的情况,

    即用最少的细绳把所有钉子连接起来。

    2、更为实际的情景是这样的情况,在某地分布着N个村庄,现在需要在N个村庄之间修路,每个村庄之前的距离不同,问怎么修最短的路,将各个村庄连接起来。


    以上这些问题都可以归纳为最小生成树问题,用正式的表述方法描述为:给定一个无方向的带权图G=(V, E),最小生成树为集合TT是以最小代价连接V中所有顶点所用边E的最小集合。 集合T中的边能够形成一颗树,这是因为每个节点(除了根节点)都能向上找到它的一个父节点。

     

    一、kruskal(克鲁斯卡尔)

    先对所有边进行排序,以权值最小的边所在的点为根节点开始处理,用一个for循遍历所有排序后的边,若这条边的两个点的根节点不同,累加上权值,再把这两个点合并,一直处理到最后即可
    需要用到并查集知识(并查集的加边操作记得用路径压缩,避免超时),和结构体的排序

    模板:

    int p[100005],r[100005];
    int n,ans;
    struct node
    {
      int x;//x,y是坐标,v是权值
      int y;
      int v;
    }a[100005];
    bool cmp(node b,node c)
    {
      return b.v<c.v;
    }
    int find(int x)//查找元素x的老板是谁
    {
        if (x == p[x])
            return x;
        else
            return p[x] = find(p[x]);
    }
    
    void join(int x, int y)//路径压缩合并两个集合
    {
        int xRoot = find(x);
        int yRoot = find(y);
    
        if (xRoot == yRoot) //老板相同,不合并
            return;
        //cnt=cnt-1;
        if (r[xRoot] < r[yRoot]) //r[i]是元素i所在树的高度,矮树的根节点认高树的根节点做老板
            p[xRoot] = yRoot;
        else if (r[xRoot] > r[yRoot])
            p[yRoot] = xRoot;
        else
        {
            p[yRoot] = xRoot;//树高相同,做老板的树高度要加一
            r[xRoot]++;
        }
    }
    void kruskal()
    {
      for(int i=1;i<=n;i++)//初始化根节点
        p[i]=i;
      sort(a+1,a+n*(n-1)/2+1,cmp);
      for(int i=1;i<=n*(n-1)/2;i++)
      {
        if(find(a[i].x)!=find(a[i].y))
        {
          join(a[i].x,a[i].y);
          ans=ans+a[i].v;
        }
      }
    }

    二、Prime(普里姆)

    由顶点开始(可以随便找一个为顶点)形成一个点集,每次从剩余点中找一个与这个点集最近的点(权值最小的点)并加入点集,直到结束

    以下流程图转载自https://blog.csdn.net/lqcsp/article/details/14118871,谢谢博主^-^

       知道了普利姆算法的核心步骤,下面我就用图示法来演示一下工作流程,如图:



    首先,确定起始顶点。我以顶点A作为起始点。根据查找法则,与点A相邻的点有点B和点H,比较AB与AH,我们选择点B,如下图。并将点B加入到U中。



    继续下一步,此时集合U中有{A,B}两个点,再分别以这两点为起始点,根据查找法则,找到边BC(当有多条边权值相等时,可选任意一条),如下图。并将点C加入到U中。



    继续,此时集合U中有{A,B,C}三个点,根据查找法则,我们找到了符合要求的边CI,如下图。并将点I加入到U中。



    继续,此时集合U中有{A,B,C,I}四个点,根绝查找法则,找到符合要求的边CF,如下图。并将点F加入到集合U中。



    继续,依照查找法则我们找到边FG,如下图。并将点G加入到U中。



    继续,依照查找法则我们找到边GH,如下图。并将点H加入到U中。



    继续,依照查找法则我们找到边CD,如下图。并将点D加入到U中。



    继续,依照查找法则我们找到边DE,如下图。并将点E加入到U中。



    此时,满足U = V,即找到了这颗最小生成树。

    模板:

    void prim()
    {
      ans=0;
      memset(vis,0,sizeof(vis));
      for(int i=2;i<=n;i++)//初始化
        dis[i]=a[1][i];
      dis[1]=0;
      vis[1]=1;
    
      for(int i=1;i<n;i++)//最后一个点不需要处理,直接加入即可,所以不要(也不能)取等
      {
        int k=0,mn=9999999;
        for(int j=1;j<=n;j++)//找出还没有被标记的点中离起点权值最小的点
        {
          if(!vis[j]&&dis[j]<mn)
          {
            mn=dis[j];
            k=j;
          }
        }
        vis[k]=1;
        ans=ans+mn;
        for(int j=1;j<=n;j++)//更新最小值,k和起点都在处理过的集合里面,更新到起点的最小值
        {
          if(!vis[j]&&dis[j]>a[k][j])
            dis[j]=a[k][j];
        }
      }
    
    }

     模板题:hdu1233 还是畅通工程 https://www.cnblogs.com/-citywall123/p/10999949.html

     

  • 相关阅读:
    高性能Javascript 选择器API学习笔记
    Backbone学习笔记二 Events
    递归用函数、存储过程实现的效果
    用触发器实现动态新增列
    局域网自动备份删除
    游标变量用法经典
    如何区分大小写字母、全角半角
    列的分拆显示
    2005的行列转换
    批量分离和附加数据库
  • 原文地址:https://www.cnblogs.com/-citywall123/p/10999911.html
Copyright © 2011-2022 走看看