洛谷 P2147 [SDOI2008]洞穴勘测 (线段树分治)
题目链接
题解
早就想写线段树分治的题了。
对于每条边,它存在于一段时间
我们按时间来搞
我们可把一条边看做一条线段
我们可以模拟线段树操作,不断分治下去
把覆盖\(l-r\)这段时间的线段筛选出来,用并查集维护联通性,回溯时撤销操作
注意不能使用路径压缩(不能破坏树的结构,方便撤销操作)
Code
#include<bits/stdc++.h>
#define LL long long
#define RG register
using namespace std;
inline int gi() {
int f = 1, s = 0;
char c = getchar();
while (c != '-' && (c < '0' || c > '9')) c = getchar();
if (c == '-') f = -1, c = getchar();
while (c >= '0' && c <= '9') s = s*10+c-'0', c = getchar();
return f == 1 ? s : -s;
}
const int N = 10010, M = 200010;
struct question {
int time, u, v;
}q[M];
int ql, w;
struct bian {
int u, v, s, e;
};
vector<bian> g;
map<pair<int, int>, int> Mp;
int siz[N], fa[N];
inline int find(int x) {
return x == fa[x] ? x : find(fa[x]);
}
pair<int, int> stk[M];
int top;
void link(int x, int y) {//按秩合并
x = find(x); y = find(y);
if (x == y) {
stk[++top] = make_pair(0, 0);
return ;
}
if (siz[x] < siz[y]) swap(x, y);
stk[++top] = make_pair(x, y);//y接在x上
fa[y] = x;
siz[x] += siz[y];
return ;
}
void clear() {//撤销操作
int x = stk[top].first, y = stk[top--].second;
if (!x && !y) return ;
fa[y] = y;
siz[x] -= siz[y];
return ;
}
void divide(int l, int r, vector<bian> E) {
vector<bian> L, R;
int tmp = top, mid = (l + r) >> 1;
for (int i = 0; i < (int)E.size(); i++) {
if (E[i].s <= l && E[i].e >= r) link(E[i].u, E[i].v);
else {
if (E[i].s <= mid) L.push_back(E[i]);
if (E[i].e > mid) R.push_back(E[i]);
}
}
if (l == r) {
while (q[w].time == l && w <= ql) {
if (find(q[w].u) == find(q[w].v)) printf("Yes\n");
else printf("No\n");
w++;
}
if (w > ql) exit(0);
return ;
}
else divide(l, mid, L), divide(mid+1, r, R);
while (top > tmp) clear();
return ;
}
int main() {
int n = gi(), m = gi(), x, y;
char s[10];
for (int i = 1; i <= m; i++) {
cin >> s >> x >> y;
if (x > y) swap(x, y);
if (s[0] == 'C') {
if (Mp.find(make_pair(x, y)) == Mp.end())
g.push_back((bian){x, y, i, m}), Mp[make_pair(x, y)] = g.size()-1;
}
else if (s[0] == 'D') {
g[Mp[make_pair(x, y)]].e = i-1;
Mp.erase(Mp.find(make_pair(x, y)));
}
else q[++ql] = (question) {i, x, y};
}
/*for (int i = 0; i < g.size(); i++)
printf("%d %d %d %d\n", g[i].u, g[i].v, g[i].s, g[i].e);*/
w = 1;
for (int i = 1; i <= n; i++) fa[i] = i, siz[i] = 1;
divide(1, m, g);
return 0;
}
洛谷 P2147 [SDOI2008]洞穴勘测 (线段树分治)的更多相关文章
- 洛谷 P2147 [SDOI2008]洞穴勘测
以下这个做法应该是叫线段树分治... 根据修改操作预处理出每条边存在的时间区间[l,r](以操作序号为时间),然后把所有形式化后的修改挂到线段树节点上. 处理完修改后,dfs一遍线段树,进入某个节点时 ...
- 洛谷P2147[SDOI2008]洞穴勘测(lct)
题目描述 辉辉热衷于洞穴勘测. 某天,他按照地图来到了一片被标记为JSZX的洞穴群地区.经过初步勘测,辉辉发现这片区域由n个洞穴(分别编号为1到n)以及若干通道组成,并且每条通道连接了恰好两个洞穴.假 ...
- [洛谷P2147][SDOI2008]洞穴勘测
题目大意:有$n$个洞穴,$m$条指令,指令有三种 $Connect\;u\;v$:在$u,v$之间连一条边 $Destroy\;u\;v$:切断$u,v$之间的边 $Query\;u\;v$:询问$ ...
- 洛谷 P2147 [SDOI2008]洞穴勘测 LCT
Code: #include <cstdio> #include <algorithm> #include <string> #include <cstrin ...
- 【洛谷P2147】洞穴勘测
题目大意:维护 N 个点的无向图,支持动态加边和删边,回答两点的连通性. 题解:线段树分治 + 可撤销并查集 询问可以离线,这是线段树分治的基础. 建立在操作时间轴上的线段树称为线段树分治算法. 本题 ...
- 洛谷P2147 [SDOI2008] 洞穴勘探 [LCT]
题目传送门 洞穴勘探 题目描述 辉辉热衷于洞穴勘测. 某天,他按照地图来到了一片被标记为JSZX的洞穴群地区.经过初步勘测,辉辉发现这片区域由n个洞穴(分别编号为1到n)以及若干通道组成,并且每条通道 ...
- P2147 [SDOI2008]洞穴勘测(LCT)
P2147 [SDOI2008]洞穴勘测 裸的LCT. #include<iostream> #include<cstdio> #include<cstring> ...
- P2147 [SDOI2008]洞穴勘测
P2147 [SDOI2008]洞穴勘测 思路 没办法,我就是喜欢板子都想发的人 都是基础操作,不多说了 代码 #include <bits/stdc++.h> #define ls ch ...
- 洛谷 P3373 【模板】线段树 2
洛谷 P3373 [模板]线段树 2 洛谷传送门 题目描述 如题,已知一个数列,你需要进行下面三种操作: 将某区间每一个数乘上 xx 将某区间每一个数加上 xx 求出某区间每一个数的和 输入格式 第一 ...
随机推荐
- MySql 之 FIND_IN_SET 和IN
CREATE TABLE `test` ( `id` int(8) NOT NULL auto_increment, `name` varchar(255) NOT NULL, `list ...
- 生成ico格式图标
ico格式可参考如下链接: http://msdn.microsoft.com/en-us/library/ms997538.aspx http://en.wikipedia.org/wiki/ICO ...
- JAVA环境安装配置
dk1.6 64位是 Java 语言的软件开发工具包,主要用于移动设备.嵌入式设备上的java应用程序. jdk1.6 64位安装教程 jdk1.6 64位JDK的安装路径:D:\Program Fi ...
- opennebula kvm日志
Fri Jul :: [InM][I]: Command execution fail: 'if [ -x "/home/oneadmin/tmp/one/im/run_probes&quo ...
- rosbag数据记录及转换图片、视频
博客转载自: https://blog.csdn.net/u012706484/article/details/78495896 rosbag常见的使用参数和配置 1.查看.bag中包含的信息 : r ...
- Makefile 编写规则 - 1
Makefilen内容 1. 显示规则:显示规则说明了,如何生成一个或多个目标.这是由Makefile指出要生成的文件和文件依赖的文件.2. 隐晦规则:基于Makefile的自动推导功能3. 变量的定 ...
- PS插件开发plugin
Photoshop插件开发 VC++制作Photoshop自动化插件:http://blog.sina.com.cn/s/blog_73c52fda0101c7hw.html Photoshop 的扩 ...
- 十进制--->二进制(利用C++栈功能)
原创 十进制转二进制很简单,其中用到C++的栈功能就能更加方便! stack<int> s; //栈的定义,s已经被定义为一个栈 s.push(); //将20入栈 s.push(); s ...
- MongoDB整理笔记のCRUD
添加 下面我们来建立一个test 的集合并写入一些数据.建立两个对象j 和t , 并保存到集合中去.在例子里 “>” 来表示是 shell 输入提示符 > j = { name : ...
- xampp本地服务器+HBuilder配置php环境
HBuilder配置PHP环境: 下载,运行HBuilder编辑器 打开右侧小窗口,点击设置图标—>设置web服务器—>外置web服务器 输入你想要浏 ...