[luoguP2147] [SDOI2008]Cave 洞穴勘测(并查集 || lct)
1.并查集骗分(数据太水,比正解还快。。。)
我们知道,并查集有一步操作叫“路径压缩”,但是本题的并查集我们不能路径压缩,否则就无法进行Destroy操作。那每一步操作我们应该怎么做呢?
对于Connect x y操作,先把x变成集合的根,之后root[x] = y;
对于Destroy x y操作,先把x变成集合的根,此时root[y]必然为x,令root[y] = y即可。
对于Query x y操作,看看x和y所在集合的根是不是一样的就好了。
那么如何把x变成集合的根呢?只要把从x到根路径上的每一条边反向即可,所以不能进行路径压缩。
其实并查集的解法也有用 lct 的思想的。
在这里,并查集中的两点之间的边就表示连接两个洞穴之间的边,非常的直接。。
注意一个细节 : 题目中说——无论通道怎么改变,任意时刻任意两个洞穴之间至多只有一条路径
也就是说不会有环!这也正是能用并查集做的原因之一。
——代码
- #include <cstdio>
- #include <iostream>
- #define N 10001
- int n, m;
- int f[N];
- inline int read()
- {
- int x = , f = ;
- char ch = getchar();
- for(; !isdigit(ch); ch = getchar()) if(ch == '-') f = -;
- for(; isdigit(ch); ch = getchar()) x = (x << ) + (x << ) + ch - '';
- return x * f;
- }
- inline int find(int x)
- {
- while(x ^ f[x]) x = f[x];
- return x;
- }
- inline void make_root(int x, int c)
- {
- if(x ^ f[x]) make_root(f[x], x);
- f[x] = c;
- }
- int main()
- {
- int i, j, x, y, fx, fy;
- char s[];
- n = read();
- m = read();
- for(i = ; i <= n; i++) f[i] = i;
- for(i = ; i <= m; i++)
- {
- scanf("%s", s);
- x = read();
- y = read();
- if(s[] == 'Q') find(x) == find(y) ? puts("Yes") : puts("No");
- else if(s[] == 'C') make_root(x, x), f[x] = y;
- else make_root(x, x), f[y] = y;
- }
- return ;
- }
2.lct(正解)
就是模板啦
——代码
- #include <cstdio>
- #include <iostream>
- #define N 10001
- #define swap(x, y) ((x) ^= (y) ^= (x) ^= (y))
- int n, m;
- int f[N], rev[N], son[N][], s[N], size[N];
- inline int read()
- {
- int x = , f = ;
- char ch = getchar();
- for(; !isdigit(ch); ch = getchar()) if(ch == '-') f = -;
- for(; isdigit(ch); ch = getchar()) x = (x << ) + (x << ) + ch - '';
- return x * f;
- }
- inline bool isroot(int x)
- {
- return son[f[x]][] ^ x && son[f[x]][] ^ x;
- }
- inline int get(int x)
- {
- return son[f[x]][] == x;
- }
- inline void pushdown(int x)
- {
- if(x && rev[x])
- {
- swap(son[x][], son[x][]);
- if(son[x][]) rev[son[x][]] ^= ;
- if(son[x][]) rev[son[x][]] ^= ;
- rev[x] = ;
- }
- }
- inline void rotate(int x)
- {
- int old = f[x], oldf = f[old], wh = get(x);
- if(!isroot(old))
- son[oldf][son[oldf][] == old] = x;
- f[x] = oldf;
- son[old][wh] = son[x][wh ^ ];
- f[son[old][wh]] = old;
- son[x][wh ^ ] = old;
- f[old] = x;
- }
- inline void splay(int x)
- {
- int i, fa, t = ;
- s[++t] = x;
- for(i = x; !isroot(i); i = f[i]) s[++t] = f[i];
- for(i = t; i >= ; i--) pushdown(s[i]);
- for(; !isroot(x); rotate(x))
- if(!isroot(fa = f[x]))
- rotate(get(x) == get(fa) ? fa : x);
- }
- inline void access(int x)
- {
- for(int t = ; x; t = x, x = f[x]) splay(x), son[x][] = t;
- }
- inline void reverse(int x)
- {
- access(x);
- splay(x);
- rev[x] ^= ;
- }
- inline int find(int x)
- {
- access(x);
- splay(x);
- while(son[x][]) x = son[x][];
- return x;
- }
- inline void link(int x, int y)
- {
- reverse(x);
- f[x] = y;
- splay(x);
- }
- inline void cut(int x, int y)
- {
- reverse(x);
- access(y);
- splay(y);
- son[y][] = f[x] = ;
- }
- int main()
- {
- int i, j, x, y;
- char s[];
- n = read();
- m = read();
- for(i = ; i <= m; i++)
- {
- scanf("%s", s);
- x = read();
- y = read();
- if(s[] == 'Q') find(x) == find(y) ? puts("Yes") : puts("No");
- if(s[] == 'C') link(x, y);
- if(s[] == 'D') cut(x, y);
- }
- return ;
- }
[luoguP2147] [SDOI2008]Cave 洞穴勘测(并查集 || lct)的更多相关文章
- 【bzoj2049】[Sdoi2008]Cave 洞穴勘测——线段树上bfs求可撤销并查集
题面 2049: [Sdoi2008]Cave 洞穴勘测 Time Limit: 10 Sec Memory Limit: 259 MB Submit: 12030 Solved: 6024 Desc ...
- 【bzoj2049】[Sdoi2008]Cave 洞穴勘测 link-cut-tree
2016-05-30 11:04:51 学习了link-cut-tree 二中神犇封禹的讲义感觉讲的超级清晰易懂啊(没有的可以q窝 算是模板吧 #include<bits/stdc++.h&g ...
- BZOJ 2049: [Sdoi2008]Cave 洞穴勘测 LCT
2049: [Sdoi2008]Cave 洞穴勘测 Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnli ...
- bzoj 2049: [Sdoi2008]Cave 洞穴勘测 动态树
2049: [Sdoi2008]Cave 洞穴勘测 Time Limit: 10 Sec Memory Limit: 259 MBSubmit: 3119 Solved: 1399[Submit] ...
- 【LCT】BZOJ2049 [SDOI2008]Cave 洞穴勘测
2049: [Sdoi2008]Cave 洞穴勘测 Time Limit: 10 Sec Memory Limit: 259 MBSubmit: 10059 Solved: 4863[Submit ...
- BZOJ_2049_[Sdoi2008]Cave 洞穴勘测_LCT
BZOJ_2049_[Sdoi2008]Cave 洞穴勘测_LCT Description 辉辉热衷于洞穴勘测.某天,他按照地图来到了一片被标记为JSZX的洞穴群地区.经过初步勘测,辉辉发现这片区域由 ...
- bzoj 2049: [Sdoi2008]Cave 洞穴勘测 (LCT)
链接:https://www.lydsy.com/JudgeOnline/problem.php?id=2049 题面: 2049: [Sdoi2008]Cave 洞穴勘测 Time Limit: 1 ...
- BZOJ 2049: [Sdoi2008]Cave 洞穴勘测 (动态树入门)
2049: [Sdoi2008]Cave 洞穴勘测 Time Limit: 10 Sec Memory Limit: 259 MBSubmit: 1528 Solved: 644[Submit][ ...
- 2049: [Sdoi2008]Cave 洞穴勘测
2049: [Sdoi2008]Cave 洞穴勘测 Time Limit: 10 Sec Memory Limit: 259 MB Submit: 7475 Solved: 3499 [Submi ...
随机推荐
- Wannafly Union Goodbye 2016-A//初识随机化~
想来想去还是把这个题写下来了.自己在补题遇到了许多问题. 给出n(n<=1e5)个点,求是否存在多于p(p>=20)×n/100的点在一条直线上... 时限20s,多组数据,暴力至少n^2 ...
- 批量格式化json
单个文件格式化工具: vscode和sublime都有格式化json的插件. 但是笔者遇到的情况是有几百个文件需要格式化,不可能每个文件都打开,进行分别格式化. 于是写了一个python脚本,非常强大 ...
- SpringMVC的controller层的方法返回值
1.ModelAndView 既带着数据,又返回视图路劲 2.String 返回试图路径 model带数据 (官方或企业推荐使用此种方式 ,此方法符合解耦思想,即数据,视图,分离 MVC) 3. ...
- NodeJS基础入门-Event
大多数Node.js核心API都采用惯用的异步事件驱动架构,其中某些类型的对象(触发器)会周期性地触发命名事件来调用函数对象(监听器). 例如,net.Server对象会在每次有新连接时触发事件;fs ...
- php读取不到https的域名
因测试环境php遇到无法正常读取到https的域名,但是域名配置了ssl证书,故做如下排查. php测试代码如下 $config['base_url'] = ''; #开启调试模式 #echo &qu ...
- 序列内置方法详解(string/list/tuple)
一.常用方法集合 1.1.string,字符串常用方法 以下举例是python2.7测试: 函数名称 作用 举例 str.capitalize() 字符串第一个字符如果是字母,则把字母替换为大写字母. ...
- pandas时间数据的集成处理
工作中遇到的一个问题: 统计各地区新能源汽车的充电时长 数据来源是北理新源的单日全球的运行数据. 这里仅统计北上广重庆四个地区的 数据处理的代码就省略了 需要整理好的是4个dataframe(数据已保 ...
- sql执行过长,如何入手优化
一条sql执行过长的时间,你如何优化,从哪些方面 1.查看sql是否涉及多表的联表或者子查询,如果有,看是否能进行业务拆分,相关字段冗余或者合并成临时表(业务和算法的优化)2.涉及链表的查询,是否能进 ...
- Linux下的硬件驱动——USB设备(转载)
usb_bulk_msg函数 当对usb设备进行一次读或者写时,usb_bulk_msg 函数是非常有用的; 然而, 当你需要连续地对设备进行读/写时,建议你建立一个自己的urbs,同时将urbs 提 ...
- HDU - 4027 Can you answer these queries?(线段树)
给定一个长度为n的序列,m次操作. 每次操作 可以将一个区间内的所有数字变为它的根号. 可以查询一个区间内所有元素的和. 线段树的初级应用. 如果把一个区间内的元素都改为它的根号的话,是需要每个数字都 ...