适合对并查集有一定理解的人.  新手可能看不懂吧....

并查集简单点说就是将相关的2个数字联系起来

比如

房子                      1   2    3   4  5   6

能通向的房子        2   3    4  5  6    1

主要 建立并查集的 函数结构 模板(一般不变除非加权--最好能理解)


for(int i=0;i<n;i++)
         flag[i]=i;               //标记数组初始化以方便寻根

 1 int find(int x)        X相当于1
{
int r=x,t; r代表的是根节点
while(x!=flag[x]) 如果 没有找到 就继续追到根为止
x=flag[x];
while(r!=x)
{ //这里是相关优化 有个定义叫做秩就是类似于楼的层数 秩越低 你就越容易从楼顶走到楼底 将秩全部变成1即一层楼
t=flag[r];
flag[r]=x;
r=t;
}
return x;
}

主要就是这个

最小生成树中  keruskai()

核心思想: 将所有元素按照某一标志  排序  然后从最前开始一次检索寻根如果

二者没有相同的根就说明他们是第一次相互关联 然后把它们的权加到value上

举例  :      你要把3个房间打通来让  别人    可以去任意房间..

这些房间没有房顶你能飞到任意房间去砸墙  但是他们的关联情况不同并且 花费的费用也不同

你要使他们的费用最低

1.  A  B 10万元
2. B C 20万元
3. A C 30万元 你可以
1,2 30
1,3 40
2,3 50
显然选择 1,2
就用上面的kruskal
排序后 先是 a-b 判断是否有相同根(打通?)没有 就打通 10万元 然后
b-c 判断是否有相同的根 没有就打通 然后继续判断
a-c 判断已经打通了 跳过...
(当然也可以加一个sum统计打通情况3个房间 2个通道就行 打通一次sum++;当sum=n-1直接跳过节省时间)
最小生成树中prime();
简单点说就是用一个二维数组记录用坐标对应 他的权(距离/价值) 然后从第第一个起点扫描和他链接的其他点
中的权值并加入一个新的数组 找到权值最小的点就是所求的,标记他,然后从他权值最小对应的点开始扫描和他相关联的点
如果比之前 存权呢个数组小就更新为新的值(重点重点重点----)具体看例题代码的
例题 poj 1861 [  http://poj.org/problem?id=1861 ]

2种解法(kruskal(克鲁斯卡尔)这个过了 ,prime 表示没过(输出感觉没有问题..)
建议..用kruskal
据说简单的快,复杂的prime快) 上代码
kruskal()
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<algorithm>
using namespace std;
int flag[],ans,p[],n,m,maxx,num;
int find(int a)
{
int r=a,tmp;
while(r!=p[r])
r=p[r];
while(a!=r)
{
tmp=p[a];
p[a]=r;
a=tmp;
}
return r;
}
struct nood
{
int x,y,w;
} pp[];
int cmp(nood xx,nood yy)
{
return xx.w<yy.w;
}
int kruskal()
{
sort(pp,pp+m,cmp);
memset(flag,,sizeof(flag));
num=maxx=;
ans=;
for(int i=; i<=n; i++)
p[i]=i;
for(int i=; i<m; i++)
{
if(num==n-)
break;
int xxx=find(pp[i].x);
int yyy=find(pp[i].y);
if(xxx!=yyy)
{
p[xxx]=yyy;
flag[ans++]=pp[i].x;
flag[ans++]=pp[i].y;
maxx=max(pp[i].w,maxx);
num++;
}
}
return ;
}
int main()
{
cin>>n>>m;
for(int i=; i<m; i++)
cin>>pp[i].x>>pp[i].y>>pp[i].w;
kruskal();
cout<<maxx<<endl<<num<<endl;
for(int i=; i<ans; i++)
{
cout<<flag[i];
if(i%==)
cout<<endl;
else
cout<<" ";
}
return ;
} /* 一种输出
1 3
1 3
2 3
2 4 输入数据
4 6
1 2 1
1 3 1
1 4 2
2 3 1
3 4 1
2 4 1 */
prime();
不知道为啥WR
代码
#include<iostream>
#include<cstring>
#include<cmath>
#include<cstdio>
#define maxx 1200
using namespace std;
int n,m,pp[maxx][maxx],value[maxx],flag[maxx],maxpath,num,a[maxx],maxxx;
int prime()
{
int ans=;
maxpath=maxxx=num=;
memset(value,,sizeof(value));
memset(a,,sizeof(a));
memset(flag,,sizeof(flag));
//初始化相关参数
for(int i=; i<=n; i++)//以 1 为起始点更新权值数组
{
if(pp[][i]!=maxx)
value[i]=pp[][i];
else
value[i]=maxx;
flag[i]=;
}
flag[]=;//标记用过了..
//寻找剩下的 边
for(int i=; i<=n; i++)
{
int k,min=maxx;//k作为下次扫描的点
for(int j=; j<=n; j++) //依次扫描寻找与k即i相关联的 最小权值
{
if(flag[j]!=&&min>value[j])
{
min=value[j];
k=j;
}
}
//更新相关参数
maxpath=max(maxpath,value[k]);
maxxx+=value[k];
a[ans++]=flag[k];
a[ans++]=k;
num++;
flag[k]=;//标记
//更新与上面相关的对应的权值
//(当这些权值小于上一次的就替换~重点!)
//就这个实现了当有2个出发点的时候
//首出发点 相对应的小于第二个的 选择了首出发点
for(int j=; j<=n; j++)
{
if(flag[j]!=&&pp[k][j]<value[j]/*此处有无等号*/)
{
value[j]=pp[k][j];
flag[j]=k;
}
}
}
cout<<maxpath<<endl<<maxxx<<endl;
for(int i=; i<ans; i++)
{
cout<<a[i];
if(i%!=)
cout<<" ";
else
cout<<endl;
}
return ;
}
int main()
{
//freopen("a1.txt","r",stdin);
//freopen("a11.txt","w",stdout);
cin>>n>>m;
//读入关联信息及其权值
memset(pp,maxx,sizeof(pp));
for(int i=; i<=m; i++)
{
int x,y;
cin>>x>>y;
cin>>pp[x][y];
}
prime();
return ;
}

或许会改....有空再说吧

												

关于最小生成树(并查集)prime和kruskal的更多相关文章

  1. UVA 1395 苗条的生成树(最小生成树+并查集)

    苗条的生成树 紫书P358 这题最后坑了我20分钟,怎么想都对了啊,为什么就wa了呢,最后才发现,是并查集的编号搞错了. 题目编号从1开始,我并查集编号从0开始 = = 图论这种题真的要记住啊!!题目 ...

  2. CSP 201703-4 地铁修建【最小生成树+并查集】

    问题描述 试题编号: 201703-4 试题名称: 地铁修建 时间限制: 1.0s 内存限制: 256.0MB 问题描述: 问题描述 A市有n个交通枢纽,其中1号和n号非常重要,为了加强运输能力,A市 ...

  3. (最小生成树 并查集)P1111 修复公路 洛谷

    题目背景 A地区在地震过后,连接所有村庄的公路都造成了损坏而无法通车.政府派人修复这些公路. 题目描述 给出A地区的村庄数N,和公路数M,公路是双向的.并告诉你每条公路的连着哪两个村庄,并告诉你什么时 ...

  4. 【BZOJ4144】[AMPPZ2014]Petrol(最短路+最小生成树+并查集)

    Description 给定一个n个点.m条边的带权无向图,其中有s个点是加油站. 每辆车都有一个油量上限b,即每次行走距离不能超过b,但在加油站可以补满. q次询问,每次给出x,y,b,表示出发点是 ...

  5. bzoj 3559: [Ctsc2014]图的分割【最小生成树+并查集】

    读题两小时系列-- 在读懂题意之后,发现M(c)就是c这块最大权割边也就是的最小生成树的最大权边的权值,所以整个问题都可以在MST的过程中解决(M和c都是跟着并查集变的) 不过不是真的最小生成树,是合 ...

  6. Luogu P1195/P1892 口袋的天空/BOI团伙 【最小生成树/并查集】By cellur925

    其实这俩题挺水的,团伙拿下了一血,但是感觉还是写一下博客比较好x. 一.团伙 题目描述 1920年的芝加哥,出现了一群强盗.如果两个强盗遇上了,那么他们要么是朋友,要么是敌人.而且有一点是肯定的,就是 ...

  7. 复习--最小生成树&&并查集

    我个人比较喜欢Kruskal算法,所以就把这个方法写了一下,但过不了洛谷,70分. 思路是先全读入,再排序,一条一条加边.运用并查集. #include<iostream> #includ ...

  8. Regional Changchun Online--Travel(最小生成树&& 并查集)

    Travel Time Limit: 1500/1000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others) Total S ...

  9. UOJ14 UER #1 DZY Loves Graph(最小生成树+并查集)

    显然可以用可持久化并查集实现.考虑更简单的做法.如果没有撤销操作,用带撤销并查集暴力模拟即可,复杂度显然可以均摊.加上撤销操作,删除操作的复杂度不再能均摊,但注意到我们在删除时就可以知道他会不会被撤销 ...

随机推荐

  1. 在ABP的Web层中实现复杂请求跨域访问

    在最近的项目中,后端使用ABP,前端采用React,前后端完全分离.其中大部分接口都通过WebApi层调用,项目中未使用Session.但最后在添加一个网站的验证码验证留言功能时,使用了Session ...

  2. 统计一个数据库中,无记录的表的sql语句

    SQL Server数据库中统计无记录数的表 大家使用的时候,将sql脚本中的红色[TestDB] 换成你的目标数据库名称. /************************************ ...

  3. 调试webpack配置文件

    webpack运行在nodejs上,调试webpack就相当于调试nodejs程序.下面介绍一种通用的办法. 1.配置package.json,加一个debug. { 'scripts': { 'de ...

  4. linux中python3安装和使用

    python安装 下载python安装包和依赖环境 #自由选择python3源码包的版本https://www.python.org/ftp/python/https://www.python.org ...

  5. 2017-11-4—稳态和暂态/瞬态(对运放积分电路的思考)[待仿真]

    先直接截图了,暂态或者说瞬态都是暂时的状态,是从一个稳定态到另一个稳定态的过程. 之所以要了解这个概念是因为对于使用运放搭建的模拟PID有很多的疑惑,比如负反馈没有电阻满不满足"虚短&quo ...

  6. Nginx配置:nginx如何配置跳转fpm

    location ~ \.php(.*)$ { fastcgi_pass 127.0.0.1:9000; fastcgi_index index.php; fastcgi_split_path_inf ...

  7. html页面转成jsp页面之后样式变化的问题解决方法

    转载:https://blog.csdn.net/zeb_perfect/article/details/51172859

  8. 熟悉Linux常用命令

    cd命令:切换目录 (1)切换到目录 /usr/localcd / usrcd / local (2)去到目前的上层目录cd .. (3)回到自己的主文件夹cd ls命令:查看文件与目录 (4)查看目 ...

  9. 深入理解iostat

    前言 iostat算是比较重要的查看块设备运行状态的工具,相信大多数使用Linux的同学都用过这个工具,或者听说过这个工具.但是对于这个工具,引起的误解也是最多的,大多数人对这个工具处于朦朦胧胧的状态 ...

  10. eolinker使用初体验(一)

    1.官网 https://www.eolinker.com 2.安装测试增强插件,由于chrome安装的时候有问题,建议移步firefox浏览器,不纠结.. 3.新建一个单例测试