【BZOJ2049】 [Sdoi2008]Cave 洞穴勘测 LCT/并查集
两种方法:
1.LCT
第一次LCT,只有link-cut和询问,无限T,到COGS上找了数据,发现splay里的父亲特判出错了(MD纸张),A了,好奇的删了反转T了。。。。
- #include <iostream>
- #include <cstdio>
- #define N 10010
- using namespace std;
- inline int read()
- {char c;int ans=;while ((c=getchar())>'' || c<'');ans=c-''; while (isdigit(c=getchar())) ans=ans*+c-''; return ans;}
- struct SplayNode
- {
- SplayNode *fa,*ch[];
- bool rev;
- bool chr() {return this==fa->ch[];}
- bool check() {return this!=fa->ch[] && this!=fa->ch[];}
- void push()
- {
- if (rev)
- {
- ch[]->rev^=true;
- ch[]->rev^=true;
- swap(ch[],ch[]);
- rev=false;
- }
- }
- }*lct[N],*null;
- int n,m;
- inline void rotate(SplayNode *x)
- {
- SplayNode *r=x->fa;
- r->push(); x->push();
- //if (x==null || r==null) return;
- bool t=x->chr();
- if (!r->check()) r->fa->ch[r->chr()]=x;
- x->fa=r->fa;
- r->ch[t]=x->ch[t^true];
- r->ch[t]->fa=r;
- x->ch[t^true]=r;
- r->fa=x;
- }
- void pushdown(SplayNode *x)
- {
- if (!x->check()) pushdown(x->fa);
- x->push();
- }
- inline void splay(SplayNode *x)
- {
- pushdown(x);
- for (;!x->check();rotate(x))
- if (!x->fa->check())
- if (x->chr()==x->fa->chr()) rotate(x->fa);
- else rotate(x);
- }
- inline SplayNode *access(SplayNode *x)
- {
- SplayNode *y=null;
- for (;x!=null;y=x,x=x->fa)
- {
- splay(x);
- x->ch[]=y;
- }
- return y;
- }
- inline void makeroot(SplayNode *x)
- {
- access(x)->rev^=;
- splay(x);
- }
- inline void cut(SplayNode *x,SplayNode *y)
- {
- makeroot(x);
- access(y);
- splay(y);
- y->ch[]->fa=null;
- y->ch[]=null;
- }
- inline void link(SplayNode *x,SplayNode *y)
- {
- makeroot(x);
- x->fa=y;
- }
- inline SplayNode *findfa(SplayNode *x)
- {
- access(x);
- splay(x);
- while (x->ch[]!=null) x=x->ch[];
- return x;
- }
- int main()
- {
- n=read(); m=read();
- null=new SplayNode;
- null->fa=null->ch[]=null->ch[]=null; null->rev=false;
- char s[];
- int x,y;
- for (int i=;i<=n;i++)
- {
- lct[i]=new SplayNode;
- lct[i]->fa=lct[i]->ch[]=lct[i]->ch[]=null; lct[i]->rev=false;
- }
- for (int i=;i<=m;i++)
- {
- scanf("%s",s);
- x=read(); y=read();
- if (s[]=='C') link(lct[x],lct[y]);
- else if (s[]=='D') cut(lct[x],lct[y]);
- else findfa(lct[x])==findfa(lct[y])?(puts("Yes")):(puts("No"));
- }
- return ;
- }
2.类并查集
每次操作都把x提到根。
1.link 直接fa[x]=y;
2.cut 因为把x提到了根,所以把y断掉。
3.暴力询问。
- #include <iostream>
- #include <cstdio>
- #include <cstdlib>
- using namespace std;
- int fa[],n,m;
- inline int read() {char c;int ans=;while ((c=getchar())>'' || c<''); ans=c-''; while (isdigit(c=getchar())) ans=ans*+c-''; return ans;}
- inline void sroot(int x) {for (int c=,f=fa[x];x;f=fa[x]) {fa[x]=c; c=x; x=f;}}
- int main()
- {
- n=read(); m=read();
- char c;
- int x,y;
- for (register int i=;i<=m;i++)
- {
- for (c=getchar(); c<'A'||c>'Z'; c=getchar());
- x=read(); y=read();
- sroot(x);
- if (c=='C') fa[x]=y;
- else if (c=='D') fa[y]=;
- else
- {
- for (;y!=x && y; y=fa[y]);
- puts(x==y?"Yes":"No");
- }
- }
- return ;
- }
PS:看得iwtwiioi的模板做的,总结的也很好,安利一下。
Description
辉辉热衷于洞穴勘测。某天,他按照地图来到了一片被标记为JSZX的洞穴群地区。经过初步勘测,辉辉发现这片区域由n个洞穴(分别编号为1到n)以及若干通道组成,并且每条通道连接了恰好两个洞穴。假如两个洞穴可以通过一条或者多条通道按一定顺序连接起来,那么这两个洞穴就是连通的,按顺序连接在一起的这些通道则被称之为这两个洞穴之间的一条路径。洞穴都十分坚固无法破坏,然而通道不太稳定,时常因为外界影响而发生改变,比如,根据有关仪器的监测结果,123号洞穴和127号洞穴之间有时会出现一条通道,有时这条通道又会因为某种稀奇古怪的原因被毁。辉辉有一台监测仪器可以实时将通道的每一次改变状况在辉辉手边的终端机上显示:如果监测到洞穴u和洞穴v之间出现了一条通道,终端机上会显示一条指令 Connect u v 如果监测到洞穴u和洞穴v之间的通道被毁,终端机上会显示一条指令 Destroy u v 经过长期的艰苦卓绝的手工推算,辉辉发现一个奇怪的现象:无论通道怎么改变,任意时刻任意两个洞穴之间至多只有一条路径。因而,辉辉坚信这是由于某种本质规律的支配导致的。因而,辉辉更加夜以继日地坚守在终端机之前,试图通过通道的改变情况来研究这条本质规律。然而,终于有一天,辉辉在堆积成山的演算纸中崩溃了……他把终端机往地面一砸(终端机也足够坚固无法破坏),转而求助于你,说道:“你老兄把这程序写写吧”。辉辉希望能随时通过终端机发出指令 Query u v,向监测仪询问此时洞穴u和洞穴v是否连通。现在你要为他编写程序回答每一次询问。已知在第一条指令显示之前,JSZX洞穴群中没有任何通道存在。
Input
第一行为两个正整数n和m,分别表示洞穴的个数和终端机上出现过的指令的个数。以下m行,依次表示终端机上出现的各条指令。每行开头是一个表示指令种类的字符串s("Connect”、”Destroy”或者”Query”,区分大小写),之后有两个整数u和v (1≤u, v≤n且u≠v) 分别表示两个洞穴的编号。
Output
对每个Query指令,输出洞穴u和洞穴v是否互相连通:是输出”Yes”,否则输出”No”。(不含双引号)
Sample Input
200 5
Query 123 127
Connect 123 127
Query 123 127
Destroy 127 123
Query 123 127
样例输入2 cave.in
3 5
Connect 1 2
Connect 3 1
Query 2 3
Destroy 1 3
Query 2 3
Sample Output
No
Yes
No
样例输出2 cave.out
Yes
No
HINT
数据说明 10%的数据满足n≤1000, m≤20000 20%的数据满足n≤2000, m≤40000 30%的数据满足n≤3000, m≤60000 40%的数据满足n≤4000, m≤80000 50%的数据满足n≤5000, m≤100000 60%的数据满足n≤6000, m≤120000 70%的数据满足n≤7000, m≤140000 80%的数据满足n≤8000, m≤160000 90%的数据满足n≤9000, m≤180000 100%的数据满足n≤10000, m≤200000 保证所有Destroy指令将摧毁的是一条存在的通道本题输入、输出规模比较大,建议c\c++选手使用scanf和printf进行I\O操作以免超时
Source
【BZOJ2049】 [Sdoi2008]Cave 洞穴勘测 LCT/并查集的更多相关文章
- [BZOJ2049][Sdoi2008]Cave 洞穴勘测 LCT模板
2049: [Sdoi2008]Cave 洞穴勘测 Time Limit: 10 Sec Memory Limit: 259 MBSubmit: 9705 Solved: 4674[Submit] ...
- [BZOJ2049] [SDOI2008] Cave 洞穴勘测 (LCT)
Description 辉辉热衷于洞穴勘测.某天,他按照地图来到了一片被标记为JSZX的洞穴群地区.经过初步勘测,辉辉发现这片区域由n个洞穴(分别编号为1到n)以及若干通道组成,并且每条通道连接了恰好 ...
- [luoguP2147] [SDOI2008]Cave 洞穴勘测(并查集 || lct)
传送门 1.并查集骗分(数据太水,比正解还快...) 我们知道,并查集有一步操作叫“路径压缩”,但是本题的并查集我们不能路径压缩,否则就无法进行Destroy操作.那每一步操作我们应该怎么做呢? 对于 ...
- bzoj2049: [Sdoi2008]Cave 洞穴勘测 lct裸题
题意:三种操作一种摧毁一条边,一种链接一条边,一种查询两个点是否联通 题解:lct的link和cut即可 /********************************************** ...
- [bzoj2049][Sdoi2008]Cave 洞穴勘测——lct
Brief Description 给定一个森林,您需要支持两种操作: 链接两个节点. 断开两个节点之间的链接. Algorithm Design 对于树上的操作,我们现在已经有了树链剖分可以处理这些 ...
- BZOJ2049 SDOI2008 Cave 洞穴勘测 【LCT】
BZOJ2049 SDOI2008 Cave 洞穴勘测 Description 辉辉热衷于洞穴勘测.某天,他按照地图来到了一片被标记为JSZX的洞穴群地区.经过初步勘测,辉辉发现这片区域由n个洞穴(分 ...
- 【LCT】BZOJ2049 [SDOI2008]Cave 洞穴勘测
2049: [Sdoi2008]Cave 洞穴勘测 Time Limit: 10 Sec Memory Limit: 259 MBSubmit: 10059 Solved: 4863[Submit ...
- bzoj 2049: [Sdoi2008]Cave 洞穴勘测 (LCT)
链接:https://www.lydsy.com/JudgeOnline/problem.php?id=2049 题面: 2049: [Sdoi2008]Cave 洞穴勘测 Time Limit: 1 ...
- BZOJ 2049: [Sdoi2008]Cave 洞穴勘测 LCT
2049: [Sdoi2008]Cave 洞穴勘测 Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnli ...
随机推荐
- C# 中的Singleton模式
一般写Singleton基本都是一下这个套路 class Singleton { public static Singleton instance; private Singleton() { } p ...
- C# DateTime时间格式转换为Unix时间戳格式
double ntime=dateTimeToUnixTimestamp(DateTime.Now); long g1 = GetUnixTimestamp(); long g2 = ConvertD ...
- 微软改名部再次大显神威——ASP.NET 5改名ASP.NET Core 1.0
(此文章同时发表在本人微信公众号"dotNET每日精华文章",欢迎右边二维码来关注.) 题记:在计算机科学领域只有两件难事:缓存不可用和命名.--Phil Karlton 今天,S ...
- 管道通信,王明学learn
管道通信 一.通讯目的 1.数据传输 一个进程需要将数据发送给另一个进程. 2.资源共享 多个进程之间共享同样的资源. 3.通知事件 一个进程需要向另一个/组进程发送消息,通知它们发生了某事件. 4. ...
- java导出word的6种方式(复制来的文章)
来自: http://www.cnblogs.com/lcngu/p/5247179.html 最近做的项目,需要将一些信息导出到word中.在网上找了好多解决方案,现在将这几天的总结分享一下. 目前 ...
- Arduino101学习笔记(二)—— 一些注意的语法点
1.宏定义 2.整数常量 3.支持C++ String类 (1)String 方法 charAt() compareTo() concat() endsWith() equals() equalsIg ...
- loadrunner处理HTTP重定向请求
//place this in global.h int HttpRetCode; int i=0; char depthVal[10]; char cTransactName[2000 ...
- JavaScript设计模式——状态模式
状态和行为: 所谓对象的状态,通常指的就是对象实例的属性的值:而行为指的就是对象的功能,再具体点说,行为大多可以对应到方法上. 状态模式的功能就是分离状态的行为,通过维护状态的变化,来调用不同状态对应 ...
- DSP using MATLAB 示例Example3.4
代码: n = [-1:3]; x = [1:5]; % x(n) = {1,2,3,4,5} % * % k = 0:500; w = (pi/500)*k; % [0,pi] axis divid ...
- SU suvelan命令学习