LOJ#121. 「离线可过」动态图连通性(线段树分治)
题意
板子题,题意很清楚吧。。
Sol
很显然可以直接上LCT。。
但是这题允许离线,于是就有了一个非常巧妙的离线的做法,好像叫什么线段树分治??
此题中每条边出现的位置都可以看做是一段区间。
我们用线段树维护。线段树的每个节点维护一个vector表示覆盖了当前节点的边的存在区间
因为总的边数是$M$的,因此线段树内总的元素最多为$logM * M$,空间可以保证
输出答案的话需要最后dfs一遍
用并查集维护出当前联通的点,需要支持撤销操作。
方法是通过按秩合并保证复杂度,不带路径压缩。
这样的话每次断父亲就行了。
我看题解里面的按秩合并都是按度数合并的,我试了一下按节点大小合并,发现跑的差不多快。。
/*
线段树分治
对于维护每一个操作出现的区间
并查集维护连通性,维护的时候记录度数,按秩合并
撤销的时候把度数小的撤销掉。
*/
#include<cstdio>
#include<vector>
using namespace std;
const int MAXN = * 1e6 + ;
inline int read() {
char c = getchar(); int x = , f = ;
while(c < '' || c > '') {if(c == '-') f = -; c = getchar();}
while(c >= '' && c <= '') x = x * + c - '', c = getchar();
return x * f;
}
int N, M;
int tim[][];//边(i, j)加入的时间
int fa[MAXN];
struct Node {
int x, deg;
}S[MAXN];
struct Query {
int opt, x, y;
}Q[MAXN];
#define ls k << 1
#define rs k << 1 | 1
struct SegTree {
int l, r;
vector<int> id;
}T[MAXN];
void Build(int k, int ll, int rr) {
T[k] = (SegTree) {ll, rr};
if(ll == rr) return ;
int mid = ll + rr >> ;
Build(ls, ll, mid); Build(rs, mid + , rr);
}
void IntervalAdd(int k, int ll, int rr, int val) {
if(ll <= T[k].l && T[k].r <= rr) {T[k].id.push_back(val); return ;}
int mid = T[k].l + T[k].r >> ;
if(ll <= mid)IntervalAdd(ls, ll, rr, val);
if(rr > mid)IntervalAdd(rs, ll, rr, val);
}
int Top, inder[MAXN];
int find(int x) {
if(fa[x] == x) return x;
else return find(fa[x]);
}
void unionn(int x, int y) {
x = find(x); y = find(y);
if(x == y) return;
if(inder[x] < inder[y]) swap(x, y);
fa[y] = x;
S[++Top] = (Node) {y, inder[y]};
if(inder[x] == inder[y]) S[++Top] = (Node) {x, inder[x] = inder[x] + inder[y]};//tag
}
void Delet(int cur) {
while(Top > cur) {
Node pre = S[Top--];
fa[pre.x] = pre.x;
inder[pre.x] = pre.deg;
}
}
void dfs(int k) {
int cur = Top;
for(int i = ; i < T[k].id.size(); i++) unionn(Q[T[k].id[i]].x, Q[T[k].id[i]].y);
if(T[k].l == T[k].r) {
if(Q[T[k].l].opt == )
putchar(find(Q[T[k].l].x) == find(Q[T[k].l].y) ? 'Y' : 'N'), putchar('\n');
} else dfs(ls), dfs(rs); Delet(cur);
}
int main() {
N = read(); M = read();
for(int i = ; i <= N; i++) fa[i] = i, inder[i] = ;
Build(, , M);//按时间为下标建线段树
for(int i = ; i <= M; i++) {
int opt = read(), x = read(), y = read();
if(x > y) swap(x, y);
if(opt == ) tim[x][y] = i;
if(opt == ) IntervalAdd(, tim[x][y], i, i), tim[x][y] = ;
Q[i] = (Query) {opt, x, y};
}
for(int i = ; i <= N; i++)
for(int j = i; j <= N; j++)
if(tim[i][j])
IntervalAdd(, tim[i][j], M, tim[i][j]);
dfs();
return ;
}
LOJ#121. 「离线可过」动态图连通性(线段树分治)的更多相关文章
- LOJ 121 「离线可过」动态图连通性——LCT维护删除时间最大生成树 / 线段树分治
题目:https://loj.ac/problem/121 离线,LCT维护删除时间最大生成树即可.注意没有被删的边的删除时间是 m+1 . 回收删掉的边的节点的话,空间就可以只开 n*2 了. #i ...
- loj#121.「离线可过」动态图连通性
题面 话说#122怎么做啊 题解 我的\(\mathrm{LCT}\)水平极差,连最小生成树都快忘了,赶紧复习一下 做法和这篇是一样的 这道题还可以练习线段树分治 还可以练习ETT 果然是道吼题 代码 ...
- LOJ #121. 「离线可过」动态图连通性 LCT维护最大生成树
这个还是比较好理解的. 你考虑如果所有边构成一棵树的话直接用 LCT 模拟一波操作就行. 但是可能会出现环,于是我们就将插入/删除操作按照时间排序,然后依次进行. 那么,我们就要对我们维护的生成树改变 ...
- 【LOJ】#121. 「离线可过」动态图连通性
题解 和BZOJ4025挺像的 就是维护边权是时间的最大生成树 删边直接删 两点未联通时直接相连,两点联通则找两点间边权小的一条边删除即可 代码 #include <bits/stdc++.h& ...
- 【LOJ121】「离线可过」动态图连通性
[LOJ121]「离线可过」动态图连通性 题面 LOJ 题解 线段树分治的经典应用 可以发现每个边出现的时间是一个区间 而我们每个询问是一个点 所以我们将所有边的区间打到一颗线段树上面去 询问每个叶子 ...
- LOJ121 「离线可过」动态图连通性
思路 动态图连通性的板子,可惜我不会在线算法 离线可以使用线段树分治,每个边按照存在的时间插入线段树的对应节点中,最后再dfs一下求出解即可,注意并查集按秩合并可以支持撤销操作 由于大量使用STL跑的 ...
- 「LOJ 121」「离线可过」动态图连通性「按时间分治 」「并查集」
题意 你要维护一张\(n\)个点的无向简单图.你被要求执行\(m\)条操作,加入删除一条边及查询两个点是否连通. 0:加入一条边.保证它不存在. 1:删除一条边.保证它存在. 2:查询两个点是否联通. ...
- LOJ.121.[离线可过]动态图连通性(线段树分治 按秩合并)
题目链接 以时间为下标建线段树.线段树每个节点开个vector. 对每条边在其出现时间内加入线段树,即,把这条边按时间放在线段树的对应区间上,会影响\(O(\log n)\)个节点. 询问就放在线段树 ...
- 「POI2011 R2 Day2」Tree Rotations【线段树合并】
题目链接 [BZOJ] [洛谷] [LOJ] 题解 由于是前序遍历,那么讨论一棵树上的逆序对的情况. 两个节点都在左子树上 两个节点都在右子树上 两个节点分别在不同的子树上. 前两种情况其实也可以归结 ...
随机推荐
- ADB运行框架原理解析【转】
本文转载自:http://blog.csdn.net/wlwl0071986/article/details/50935496 一.adb守护进程的初始化 源码路径:~/system/core/adb ...
- javascript 无刷新上传图片之原理
刚开始我认为可以像ajax 那样获取到数据然后通过ajax 发送请求,后来发现浏览器为了客户端的安全默认并没有给javascript 这个权限.这个方法当然是行不同了.我看了好像开源的上传图片原理,当 ...
- Silverlight结合Web Service进行文件上传
search了非常多的文章,总算勉强实现了.有许多不完善的地方. 在HCLoad.Web项目下新建目录Pics复制一张图片到根目录下. 图片名:Bubble.jpg 右击->属性->生成操 ...
- JQ的双向数据绑定
把渲染页面封装在function里面 在修改后直接调用,但是如果有alert();一定要注意function的摆放位置,正常应该是在alert前面,应为如果在后面会影响重新渲染的效果
- python(一):multiprocessing——死锁
前言近年来,使用python的人越来越多,这得益于其清晰的语法.低廉的入门代价等因素.尽管python受到的关注日益增多,但python并非完美,例如被人诟病最多的GIL(值得注意的是,GIL并非py ...
- js动态改变img元素src在IE无效的问题
做了个验证码功能,需要做个点击改变验证码图片的功能,使用js改变img的src,代码如下 $("#cerificationCodeImg").attr("src" ...
- (二十五)后台开发-分类信息的curd -展示所有实现
案例1-分类信息的curd 步骤分析: 左边的dtree: 1.导入dtree.js 2.导入dtree.css 3.创建一个div 添加样式 class="dtree" 4.在d ...
- CodeForces 712B Memory and Trident (水题,暴力)
题意:给定一个序列表示飞机要向哪个方向飞一个单位,让你改最少的方向,使得回到原点. 析:一个很简单的题,把最后的位置记录一下,然后要改的就是横坐标和纵坐标绝对值之和的一半. 代码如下: #pragma ...
- bzoj 3160: 万径人踪灭【FFT+manacher】
考虑正难则反,我们计算所有对称子序列个数,再减去连续的 这里减去连续的很简单,manacher即可 然后考虑总的,注意到关于一个中心对称的两点下标和相同(这样也能包含以空位为对称中心的方案),所以设f ...
- 【CodeForces - 651C 】Watchmen(map)
Watchmen 直接上中文 Descriptions: 钟表匠们的好基友马医生和蛋蛋现在要执行拯救表匠们的任务.在平面内一共有n个表匠,第i个表匠的位置为(xi, yi). 他们需要安排一个任务计划 ...