「长乐集训 2017 Day1」区间 线段树
题目
对于两个区间\((a,b),(c,d)\),若\(c < a < d\)或\(c < b < d\)则可以从\((a,b)\)走到\((c,d)\)去,现在有以下两种操作:
- 给定\(1 \space x \space y\),表示在集合中添加\((x,y)\)这个区间,保证新加入的这个区间一定比之前的所有区间长度长。
- 给定\(2 \space a \space b\),表示询问是否有一条路径能从第\(a\)个区间走到第\(b\)个区间。
初始时区间集合为空,现在请你回答所有的询问
\(1 \leq n \leq 10^5,所有数字绝对值 \leq 10^9\)
题解:
容易发现新加入的区间与其左右端点落到的区间一定是可以互达的。
所以可以合并其集合。
那么对于新加入的区间所覆盖的区间来说,多了一条到新加入区间的单向边。
我们可以统计记录所有集合的最小左端点和最大右端点来判断一个区间是否可以通过有向边到达另一个区间。
对于区间的维护我们可以使用并查集。
同时使用线段树来维护插入区间和查询覆盖单点的所有区间的操作。
由于每次进行完合并操作后点上的所有的区间都会合并为一个。
所以总体复杂度\(O(n\log^2n)\)
#include <vector>
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long ll;
inline void read(int &x){
x=0;static char ch;static bool flag;flag = false;
while(ch=getchar(),ch<'!');if(ch == '-') ch=getchar(),flag = true;
while(x=10*x+ch-'0',ch=getchar(),ch>'!');if(flag) x=-x;
}
#define rg register int
#define rep(i,a,b) for(rg i=(a);i<=(b);++i)
#define per(i,a,b) for(rg i=(a);i>=(b);--i)
const int maxn = 100010;
int L[maxn],R[maxn],op[maxn];
int c[maxn<<1],cnt = 0,fa[maxn];
int find(int x){return fa[x] == x ? x : fa[x] = find(fa[x]);}
inline void Union(int u,int v){
int x = find(u),y = find(v);
if(x == y) return ;
fa[x] = y;
L[y] = min(L[y],L[x]);
R[y] = max(R[y],R[x]);
return ;
}
vector<int>ve[maxn<<3];
void modify(int rt,int l,int r,int p,int id){
for(vector<int>::iterator it = ve[rt].begin();it != ve[rt].end();++it) Union(*it,id);
if(ve[rt].empty() == false) ve[rt].clear(),ve[rt].push_back(find(id));
if(l == r) return ;
int mid = l+r >> 1;
if(p <= mid) modify(rt<<1,l,mid,p,id);
else modify(rt<<1|1,mid+1,r,p,id);
}
void insert(int rt,int l,int r,int L,int R,int id){
if(L <= l && r <= R){ve[rt].push_back(id);return ;}
int mid = l+r >> 1;
if(L <= mid) insert(rt<<1,l,mid,L,R,id);
if(R > mid) insert(rt<<1|1,mid+1,r,L,R,id);
}
int idx[maxn],num = 0;
int main(){
freopen("interval.in","r",stdin);
freopen("interval.out","w",stdout);
int n;read(n);
rep(i,1,n){
read(op[i]);read(L[i]);read(R[i]);
if(op[i] == 1) c[++cnt] = L[i],c[++cnt] = R[i];
}
sort(c+1,c+cnt+1);
rep(i,1,n){
if(op[i] == 1){
int l = lower_bound(c+1,c+cnt+1,L[i]) - c;
int r = lower_bound(c+1,c+cnt+1,R[i]) - c;
idx[++ num] = i;fa[i] = i;
modify(1,1,cnt,l,i);modify(1,1,cnt,r,i);
if(l+1 <= r-1) insert(1,1,cnt,l+1,r-1,i);
}else{
int u = find(idx[L[i]]);
int v = find(idx[R[i]]);
//printf("u = %d,v = %d\n",u,v);
if(u == v) puts("YES");
else if(L[v] < L[u] && L[u] < R[v]) puts("YES");
else if(L[v] < R[u] && R[u] < R[v]) puts("YES");
else puts("NO");
}
}
return 0;
}
「长乐集训 2017 Day1」区间 线段树的更多相关文章
- LOJ #6029. 「雅礼集训 2017 Day1」市场 线段树维护区间除法
题目描述 从前有一个贸易市场,在一位执政官到来之前都是非常繁荣的,自从他来了之后,发布了一系列奇怪的政令,导致贸易市场的衰落. 有 \(n\) 个商贩,从\(0 \sim n - 1\) 编号,每个商 ...
- 【loj6029】「雅礼集训 2017 Day1」市场 线段树+均摊分析
题目描述 给出一个长度为 $n$ 的序列,支持 $m$ 次操作,操作有四种:区间加.区间下取整除.区间求最小值.区间求和. $n\le 100000$ ,每次加的数在 $[-10^4,10^4]$ 之 ...
- 「雅礼集训 2017 Day1」市场 (线段树除法,区间最小,区间查询)
老师说,你们暴力求除法也整不了多少次就归一了,暴力就好了(应该只有log(n)次) 于是暴力啊暴力,结果我归天了. 好吧,在各种题解的摧残下,我终于出了一篇巨好看(chou lou)代码(很多结构体党 ...
- loj#6029. 「雅礼集训 2017 Day1」市场(线段树)
题意 链接 Sol 势能分析. 除法是不能打标记的,所以只能暴力递归.这里我们加一个剪枝:如果区间内最大最小值的改变量都相同的话,就变成区间减. 这样复杂度是\((n + mlogn) logV\)的 ...
- #6029. 「雅礼集训 2017 Day1」市场 [线段树]
考虑到每次除法,然后加法,差距会变小,于是维护加法lazytag即可 #include <cstdio> #include <cmath> #define int long l ...
- 「长乐集训 2017 Day10」划分序列 (二分 dp)
「长乐集训 2017 Day10」划分序列 题目描述 给定一个长度为 n nn 的序列 Ai A_iAi,现在要求把这个序列分成恰好 K KK 段,(每一段是一个连续子序列,且每个元素恰好属于一 ...
- loj6271 「长乐集训 2017 Day10」生成树求和 加强版(矩阵树定理,循环卷积)
loj6271 「长乐集训 2017 Day10」生成树求和 加强版(矩阵树定理,循环卷积) loj 题解时间 首先想到先分开三进制下每一位,然后每一位分别求结果为0,1,2的树的个数. 然后考虑矩阵 ...
- loj #6032. 「雅礼集训 2017 Day2」水箱 线段树优化DP转移
$ \color{#0066ff}{ 题目描述 }$ 给出一个长度为 \(n\) 宽度为 \(1\) ,高度无限的水箱,有 \(n-1\) 个挡板将其分为 \(n\) 个 \(1 - 1\) 的小格, ...
- loj6271「长乐集训 2017 Day10」生成树求和 加强版
又是一个矩阵树套多项式的好题. 这里我们可以对每一位单独做矩阵树,但是矩阵树求的是边权积的和,而这里我们是要求加法,于是我们i将加法转化为多项式的乘法,其实这里相当于一个生成函数?之后如果我们暴力做的 ...
随机推荐
- 字典,字符串,元组,字典,集合set,类的初步认识,深浅拷贝
Python之路[第二篇]:Python基础(一) 入门知识拾遗 一.作用域 对于变量的作用域,执行声明并在内存中存在,该变量就可以在下面的代码中使用. if 1==1: name = 'Jaso ...
- 8.22 ps课堂练习
真是做得超烂!以前学的快忘光了!
- 《Effective MySQL之SQL语句最优化》读书笔记——乱七八糟系列(给自己看)
该书区别于诸如<MySQL技术内幕——InnoDB存储引擎>等书的一大特色是该书主要讲的是MySQL数据库中的索引技术,并分别讲了InnoDB.MyISAM.Memory三个存储引擎.其中 ...
- matlab fread
Matlab中fread函数用法 “fread”以二进制形式,从文件读出数据. 语法1:[a,count]=fread(fid,size,precision) 语法2:[a,count]=fre ...
- ggplot2学习总结
- 【Tech】SQL Server2008使用笔记
1.64位win7系统报错“未在本地计算机上注册 Microsoft.ACE.OLEDB.12.0 ”. 解决方法:从这个网址下载和安装一个驱动http://www.microsoft.com/zh- ...
- JavaScript笔记02——基本语法(包括函数、对象、数组等)
Doing Math & Logic Conditional & Looping Functions Objects Arrays Doing Math & Logic 1.J ...
- Linux网络检测手段汇总
1.iftop iftop可测量通过每一个套接字连接传输的数据:它采用的工作方式有别于nload.iftop使用pcap库来捕获进出网络适配器的数据包,然后汇总数据包大小和数量,搞清楚总的带宽使用情况 ...
- INDEL的重新比对和碱基质量分数的重新校准
1.为什么要做这两步(why): indel的重新比对:这是由于比对软件的自身限制,其可能将包括indel的read解释为snp的read,这就导致calling的错误和后面的碱基质量分数的重新校准. ...
- css transform常用变化解析
本文旨在对常用变化做最直观的简析 translate 移动 translateX() X轴正方向移动(单位可为px,也可为%,为%时以自身为参照物) translateY() Y轴反方向移动 tran ...