洛谷 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 求出某区间每一个数的和 输入格式 第一 ...
随机推荐
- 665. Non-decreasing Array只允许修改一位数的非递减数组
[抄题]: Given an array with n integers, your task is to check if it could become non-decreasing by mod ...
- 118. Pascal's Triangle杨辉三角形(全部/一行)
[抄题]: Given numRows, generate the first numRows of Pascal's triangle. For example, given numRows = 5 ...
- 5-有道爬虫demo(post)
爬取有道页面,实现中文翻译成英文: #_*_ coding: utf-8 _*_ ''' Created on 2018-7-12 @author: sss 功能:爬取有道翻译 ''' import ...
- 38-最长公共子序列(dp)
最长公共子序列 https://www.nowcoder.com/practice/c996bbb77dd447d681ec6907ccfb488a?tpId=49&&tqId=293 ...
- django 1.8 官方文档翻译:13-12 验证器
django 1.8 官方文档翻译:13-12 验证器 2015年09月20日 21:36:18 ApacheCN_飞龙 阅读数:639 https://blog.csdn.net/wizardfo ...
- 正则表达式复习 (?<=) (?=)
1.首先值得一说的是"<" 和">" 不是元字符 "."是元字符 ,连接字符"-",即使在字符组内部也不一定 ...
- 第十课,ROS仿真2
Rviz 属性 1.下面以turtlebot_stage inRviz为例 首先安装 sudo apt-get install ros-indigo-turtlebot-simulator 运行 ro ...
- (转)15个非常棒的jQuery无限滚动插件【瀑布流效果】
原文地址:http://www.cnblogs.com/lyw0301/archive/2013/06/19/3145084.html 现在,最热门的网站分页趋势之一是jQuery的无限滚动(也即瀑布 ...
- Windows Server 2003 asp网页不能访问的常见问题
1. [开始]--[程序]--[管理工具]--[Internet信息服务管理器],在服务器名下的“web服务扩展”的右窗口,单击active server pages -> 单击[允许].2. ...
- 通过vb.net 和NPOI实现对excel的读操作
通过vb.net 和NPOI实现对excel的读操作,很久很久前用过vb,这次朋友的代码是vb.net写的需要一个excel的操作, 就顾着着实现功能了,大家凑合着看吧 Option Explicit ...