BZOJ2049:Cave 洞穴勘测 (LCT入门)
- 辉辉热衷于洞穴勘测。某天,他按照地图来到了一片被标记为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
- 样例输入1 cave.in
- Query
- Connect
- Query
- Destroy
- Query
- 样例输入2 cave.in
- Connect
- Connect
- Query
- Destroy
- Query
- Sample Output
- 样例输出1 cave.out
- No
- Yes
- No
- 样例输出2 cave.out
- Yes
- No
题意:一个森林,有三种操作。
L,u,v: 在u和v之间加一条边。
C,u,v:删去u和v之间的边。
Q,u,v:查询u和v是否在同一棵树里。
思路:简单的LCT。
具体的:
- 关于查找两个节点X、Y是否在同一棵树上:因为splay的根节点是不固定的,也没有代表性,所以需要找“原树”的根节点。 具体的:把Xaceess,然后splay到根,然后就可以向左边找深度第的节点,直到找到原树的根节点。
- 由于有多棵splay树,所以不能仅仅记录root节点,而需要用isroot函数判定X是否是一棵树的根。所以在rotate的时候要先把X和fa[fa[X]]的关系搞定,不然会错。
- splay操作要先pushdown。
- #include<cstdio>
- #include<cstdlib>
- #include<iostream>
- #include<algorithm>
- using namespace std;
- const int maxn=;
- struct LCT
- {
- int rev[maxn],ch[maxn][],fa[maxn],stc[maxn],cnt;
- int isroot(int x){
- return ch[fa[x]][]!=x&&ch[fa[x]][]!=x;
- }
- void rotate(int x)
- {
- int old=fa[x],fold=fa[old];
- int opt=(ch[old][]==x);
- if(!isroot(old)) ch[fold][ch[fold][]==old]=x;
- fa[x]=fold; //x和fold的关系最先判定,不然会和下面冲突。
- ch[old][opt]=ch[x][opt^]; fa[ch[old][opt]]=old;
- ch[x][opt^]=old; fa[old]=x;
- }
- void splay(int x)
- {
- int top=; stc[++top]=x;
- for(int i=x;!isroot(i);i=fa[i]) stc[++top]=fa[i];
- for(int i=top;i;i--) pushdown(stc[i]);
- for(int f;!isroot(x);rotate(x)){
- f=fa[x];
- if(!isroot(f))
- rotate((ch[f][]==x)==(ch[fa[f]][]==f)?f:x);
- }
- }
- void pushdown(int x)
- {
- if(!rev[x]) return ;
- int l=ch[x][],r=ch[x][];
- rev[l]^=; rev[r]^=; rev[x]=;
- swap(ch[x][],ch[x][]);
- }
- void access(int x)
- {
- int rson=;
- while(x){
- splay(x);
- ch[x][]=rson;
- rson=x; x=fa[x];
- }
- }
- int find(int x){
- access(x); splay(x);
- while(ch[x][]) x=ch[x][];
- return x;
- }
- int make_root(int x){ access(x); splay(x); rev[x]^=; }
- void link(int x,int y){ make_root(x); fa[x]=y; splay(x); }
- void cut(int x,int y){ make_root(x); access(y); splay(y); fa[x]=ch[y][]=; }
- }S;
- char ch[];
- int main()
- {
- int n,m,x,y;
- scanf("%d%d",&n,&m);
- while(m--){
- scanf("%*c%s%d%d",&ch,&x,&y);
- if(ch[]=='C') S.link(x,y);
- if(ch[]=='D') S.cut(x,y);
- if(ch[]=='Q'){
- if(S.find(x)==S.find(y)) printf("Yes\n");
- else printf("No\n");
- }
- }
- }
BZOJ2049:Cave 洞穴勘测 (LCT入门)的更多相关文章
- bzoj2049 Cave 洞穴勘测 lct
这里比上次多了几个操作. 1. make_root(u) 换根节点, 先access(u), 再splay(u),将u移动到splay树的最顶上, 现在这棵splay对于root来说 只有左子树上有东 ...
- [BZOJ2049][Sdoi2008]Cave 洞穴勘测 LCT模板
2049: [Sdoi2008]Cave 洞穴勘测 Time Limit: 10 Sec Memory Limit: 259 MBSubmit: 9705 Solved: 4674[Submit] ...
- BZOJ-2049 Cave洞穴勘测 动态树Link-Cut-Tree (并查集骗分TAT)
2049: [Sdoi2008]Cave 洞穴勘测 Time Limit: 10 Sec Memory Limit: 259 MB Submit: 5833 Solved: 2666 [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 ...
- 【BZOJ2049】 [Sdoi2008]Cave 洞穴勘测 LCT/并查集
两种方法: 1.LCT 第一次LCT,只有link-cut和询问,无限T,到COGS上找了数据,发现splay里的父亲特判出错了(MD纸张),A了,好奇的删了反转T了.... #include < ...
- [BZOJ2049] [SDOI2008] Cave 洞穴勘测 (LCT)
Description 辉辉热衷于洞穴勘测.某天,他按照地图来到了一片被标记为JSZX的洞穴群地区.经过初步勘测,辉辉发现这片区域由n个洞穴(分别编号为1到n)以及若干通道组成,并且每条通道连接了恰好 ...
- 【bzoj2049】[Sdoi2008]Cave 洞穴勘测 LCT
题目描述 辉辉热衷于洞穴勘测.某天,他按照地图来到了一片被标记为JSZX的洞穴群地区.经过初步勘测,辉辉发现这片区域由n个洞穴(分别编号为1到n)以及若干通道组成,并且每条通道连接了恰好两个洞穴.假如 ...
- BZOJ2049[Sdoi2008]洞穴勘测——LCT
题目描述 辉辉热衷于洞穴勘测.某天,他按照地图来到了一片被标记为JSZX的洞穴群地区.经过初步勘测,辉辉发现这片区域由n个洞穴(分别编号为1到n)以及若干通道组成,并且每条通道连接了恰好两个洞穴.假如 ...
- bzoj2049: [Sdoi2008]Cave 洞穴勘测 lct裸题
题意:三种操作一种摧毁一条边,一种链接一条边,一种查询两个点是否联通 题解:lct的link和cut即可 /********************************************** ...
随机推荐
- Python浅拷贝copy()与深拷贝deepcopy()区别
其实呢,浅拷贝copy()与深拷贝deepcopy()之间的区分必须要涉及到python对于数据的存储方式. 首先直接上结论: —–我们寻常意义的复制就是深复制,即将被复制对象完全再复制一遍作为独立的 ...
- POJ 1080 Human Gene Functions 【dp】
题目大意:每次给出两个碱基序列(包含ATGC的两个字符串),其中每一个碱基与另一串中碱基如果配对或者与空串对应会有一个分数(可能为负),找出一种方式使得两个序列配对的分数最大 思路:字符串动态规划的经 ...
- POJ 2420 A Star not a Tree?【爬山法】
题目大意:在二维平面上找出一个点,使它到所有给定点的距离和最小,距离定义为欧氏距离,求这个最小的距离和是多少(结果需要四舍五入)? 思路:如果不能加点,问所有点距离和的最小值那就是经典的MST,如果只 ...
- hdu 2845
#include<stdio.h> #define N 200100 int f[N]; int a[N],n; int main() { int m,j,i,suma,sumb,sum ...
- FastDfs + Dht 搭建笔记
以下为搭建一套分布式文件集群系统,参考了很多资料,自己经过在服务器上搭建并且经过了测试.记录以方便以后使用查看. FastDfs + Dht 安装手册 一:概述 FastDFS是由淘宝的余庆先生所开发 ...
- python学习之-- RabbitMQ 消息队列
记录:异步网络框架:twisted学习参考:www.cnblogs.com/alex3714/articles/5248247.html RabbitMQ 模块 <消息队列> 先说明:py ...
- P2835 刻录光盘
洛谷—— P2835 刻录光盘 题目描述 在JSOI2005夏令营快要结束的时候,很多营员提出来要把整个夏令营期间的资料刻录成一张光盘给大家,以便大家回去后继续学习.组委会觉得这个主意不错!可是组委会 ...
- babel 用法及其 .babelrc 的配置详解,想做前端架构,拒绝一知半解...
Babel 官方介绍:将 ECMAScript 2015 及其版本以后的 javascript 代码转为旧版本浏览器或者是环境中向后兼容版本的 javascript 代码. 简而言之,就是把不兼容的 ...
- crontab使用简介
crontab的配置文件: 前四行是用来配置crond任务运行的环境变量 第一行SHELL变量指定了系统要使用哪个shell,这里是bash 第二行PATH变量指定了系统执行命令的路径 第三行MAIL ...
- 【APUE】关于信号的一些常用函数
kill和raise函数 #include <signal.h> int kill(pid_t pid,int signo); int raise(int signo);//两个函数返回值 ...