部分内容摘自以下大佬的博客,感谢他们!

http://blog.csdn.net/dm_vincent/article/details/7769159

http://blog.csdn.net/dm_vincent/article/details/7655764

基础得可以去看上面的连接,再此我只记录一些温故而知新得知识点

并查集,连通起来后类似一颗颗得树,我们得基本操作是Find函数,寻找树根,所以树得高度,成为了,并查集算法快不快的关键条件,为了加快算法,我们采取以下操作:

1.尽可能得让每一个结点直接连到树根(过程中优化)

int Find(int x)
{
while(x != pre[x])
{
pre[x] = pre[pre[x]];
x = pre[x];
}
return x;
}

2.进行树得合并得时候,避免造成畸形数目

void join(int a,int b)
{
int u = Find(a);
int v = Find(b);
if(u != v)
{
if(s[u] > s[v])
{
pre[v] = u;
s[u] += s[v];
}
else
{
pre[u] = v;
s[v] += s[u];
}
}
}

这里用到了s数组记录的是这个树得大小,一开始初始化都为1,更新只需要再连树得时候进行

我觉得以下这个理解或者应用,让我更加理解了并查集得数据结构,以及解决问题时得建模思考方向

对这个问题抽象之后,就是要求进行若干次union操作之后,还会剩下多少颗树(或者说还剩下多少Connected Components)。反映到这个例子中,就是要求有多少“圈子”。其实,这也是社交网络中的最基本的功能,每次系统向你推荐的那些好友一般而言,会跟你在一个“圈子”里面,换言之,也就是你可能认识的人,以并查集的视角来看这层关系,就是你们挂在同一颗树上。

HDU3635_不错的一道题目

题目大意:

题意:起初球i是被放在i号城市的,在年代更迭,世事变迁的情况下,球被转移了,而且转移的时候,连带该城市的所有球都被移动了:T A B(A球所在的城市的所有球都被移动到了B球所在的城市),Q A(问:A球在那城市?A球所在城市有多少个球呢?A球被转移了多少次呢?)

(上面题意的描述摘自:http://www.cnblogs.com/Shirlies/archive/2012/03/06/2382118.html)

不错得一道题目,按照我们原来的并查集优化来想,前两个输出都比较简单,就是次数问题

(ps:当然了,根据题意,我们时不能进行平衡树得保持得,因为最终状态(树根)是确定得)

对于次数:第一个跟新得地方肯定是join函数里面,比较容易想到的时一旦合并,就使每一个合并的元素++,这样的话会超时,只能优化,有点类似线段树中得lazy标记,我们不能一更到底,那就先更新最上层,然后想办法传递给下层,如和传递呢,好像我们还有Find函数

在Find函数中,我们会进行路径压缩,这很好,因为保证了,tm【x】相加得唯一性,这时候要稍微改变以下Find函数,因为我要传递得值以前是根,现在不是了所以我要后退一步,就是从判断x是不是根后退到判断pre【x】是不是根,然后进行tm【x】得更新优化,一旦优化完毕我们就进行路径压缩,防止合并多次。虽然有了路径压缩,但是最后我们还是要进行一次Find(a),才能进行后续得输出。

不错的一道题目,

#include <iostream>
#include <string.h>
#include <cstdio>
using namespace std;
const int maxn = 1e6+1e3;
int pre[maxn];
int s[maxn];
int tm[maxn]; int Find(int x)
{
//路径压缩
while(pre[x] != pre[pre[x]])
{
tm[x] += tm[pre[x]];
pre[x] = pre[pre[x]];//路径压缩!!非常棒!!!
}
return pre[x];
}
void join(int a,int b)
{
int u = Find(a);
int v = Find(b);
if(u != v)
{
pre[u] = v;
tm[u]++;
s[v] += s[u];
}
}
void init(int n)
{
for(int i = 0;i <= n;i++)
{
pre[i] = i;
s[i] = 1;
tm[i] = 0;
}
}
int main()
{
char c;
int t,a,b,n,m;
scanf("%d",&t);getchar();
int cas = 1;
while(t--)
{
scanf("%d %d",&n,&m);getchar();
init(n);
printf("Case %d:\n",cas++);
while(m--)
{
scanf("%c",&c);
if(c == 'T')
{
scanf("%d %d",&a,&b);getchar();
join(a,b);
}
else
{
scanf("%d",&a);getchar();
int ans = Find(a);
printf("%d %d %d\n",ans,s[ans],tm[a]);
printf("%d && \n",pre[a]);
}
}
}
return 0;
}

【lazy标记得思想】HDU3635 详细学习并查集的更多相关文章

  1. Dragon Balls(hdu3635带权并查集)

    题意:n个城市有n个龙珠,T  a,b 代表将龙珠a移动到b城市,Q a代表查询龙珠a所在城市,该城市有多少颗龙珠,该龙珠移动了多少次. 注意:移动时是将龙珠所在该城市所有龙珠都移动,每个龙珠不会再回 ...

  2. 算法学习 并查集(Union-Find) (转)

    并查集是我暑假从高手那里学到的一招,觉得真是太精妙的设计了.以前我无法解决的一类问题竟然可以用如此简单高效的方法搞定.不分享出来真是对不起party了.(party:我靠,关我嘛事啊?我跟你很熟么?) ...

  3. HDU 1698 Just a Hook (线段树 成段更新 lazy-tag思想)

    题目链接 题意: n个挂钩,q次询问,每个挂钩可能的值为1 2 3,  初始值为1,每次询问 把从x到Y区间内的值改变为z.求最后的总的值. 分析:用val记录这一个区间的值,val == -1表示这 ...

  4. 浅谈算法——线段树之Lazy标记

    一.前言 前面我们已经知道线段树能够进行单点修改和区间查询操作(基本线段树).那么如果需要修改的是一个区间该怎么办呢?如果是暴力修改到叶子节点,复杂度即为\(O(nlog n)\),显然是十分不优秀的 ...

  5. Hbase技术详细学习笔记

    注:转自 Hbase技术详细学习笔记 最近在逐步跟进Hbase的相关工作,由于之前对Hbase并不怎么了解,因此系统地学习了下Hbase,为了加深对Hbase的理解,对相关知识点做了笔记,并在组内进行 ...

  6. 万字长文,以代码的思想去详细讲解yolov3算法的实现原理和训练过程,Visdrone数据集实战训练

    以代码的思想去详细讲解yolov3算法的实现原理和训练过程,并教使用visdrone2019数据集和自己制作数据集两种方式去训练自己的pytorch搭建的yolov3模型,吐血整理万字长文,纯属干货 ...

  7. Redis初识、设计思想与一些学习资源推荐

    一.Redis简介 1.什么是Redis Redis 是一个开源的使用ANSI C 语言编写.支持网络.可基于内存亦可持久化的日志型.Key-Value 数据库,并提供多种语言的API.从2010 年 ...

  8. HDU 3911 Black And White (线段树区间合并 + lazy标记)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3911 给你n个数0和1,m个操作: 0操作  输出l到r之间最长的连续1的个数 1操作  将l到r之间 ...

  9. POJ 3237 Tree (树链剖分 路径剖分 线段树的lazy标记)

    题目链接:http://poj.org/problem?id=3237 一棵有边权的树,有3种操作. 树链剖分+线段树lazy标记.lazy为0表示没更新区间或者区间更新了2的倍数次,1表示为更新,每 ...

随机推荐

  1. ArcGIS案例学习笔记1_1

    ArcGIS案例学习笔记1_1 联系方式:谢老师,135_4855_4328, xiexiaokui#qq.com 时间:第一天上午 准备 0.U盘复制ArcGIS培训*** 1.练习数据不要放到桌面 ...

  2. VS Code常用快捷键总结

    目录: 1.主命令框 2.常用快捷键   (1) 编辑器与窗口管理   (2) 代码编辑     <1> 格式调整     <2> 光标相关     <3> 重构代 ...

  3. 使用spring-data-JPA调用存储过程

    第一种情况,对于只有一个或没有返回值的存储过程,使用如下方式调用: @Entity @NamedStoredProcedureQuery(name = "pro1", proced ...

  4. C#aspx页面前台使用<%=%>无法取到后台的值

    检查是不是有拼接问题,正常public和protected修饰的字段或属性均可使用<%=%>.另外,加载(Page_Load)时有没有给它们赋初始值? 答 1)前台页面只能调用本后置代码的 ...

  5. 图解Java常用数据结构(一)【转载】

    最近在整理数据结构方面的知识, 系统化看了下Java中常用数据结构, 突发奇想用动画来绘制数据流转过程. 主要基于jdk8, 可能会有些特性与jdk7之前不相同, 例如LinkedList Linke ...

  6. VMware安装win7:units specified don't exist问题

    主要是磁盘接口不匹配,调整CD/DVD和硬件磁盘接口, CD/DVD调整成IDE,硬盘调整成SATA即可. 提示system not found,主分区没有激活,进入disgenius,会提示修正,保 ...

  7. SqlServer 查询死锁,杀死死锁进程*转载

    原文: -- 查询死锁 select request_session_id spid, OBJECT_NAME(resource_associated_entity_id) tableName fro ...

  8. 排序矩阵中的从小到大第k个数 · Kth Smallest Number In Sorted Matrix

    [抄题]: 在一个排序矩阵中找从小到大的第 k 个整数. 排序矩阵的定义为:每一行递增,每一列也递增. [思维问题]: 不知道应该怎么加,因为不是一维单调的. [一句话思路]: 周围两个数给x或y挪一 ...

  9. oracle基本查询入门(一)

    一.基本select语句 SELECT *|{[DISTINCT] column|expression [alias], ...} FROM table; 例如: --查询所有数据 select * ...

  10. win 下 nginx 的虚拟主机创建

    1.在nginx安装目录下的conf下创建vhost目录,用于存放虚拟主机配置文件.   2.在nginx安装目录下的conf/nginx.conf的http{}中加入 include vhost/* ...