题目

对于两个区间\((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」区间 线段树的更多相关文章

  1. LOJ #6029. 「雅礼集训 2017 Day1」市场 线段树维护区间除法

    题目描述 从前有一个贸易市场,在一位执政官到来之前都是非常繁荣的,自从他来了之后,发布了一系列奇怪的政令,导致贸易市场的衰落. 有 \(n\) 个商贩,从\(0 \sim n - 1\) 编号,每个商 ...

  2. 【loj6029】「雅礼集训 2017 Day1」市场 线段树+均摊分析

    题目描述 给出一个长度为 $n$ 的序列,支持 $m$ 次操作,操作有四种:区间加.区间下取整除.区间求最小值.区间求和. $n\le 100000$ ,每次加的数在 $[-10^4,10^4]$ 之 ...

  3. 「雅礼集训 2017 Day1」市场 (线段树除法,区间最小,区间查询)

    老师说,你们暴力求除法也整不了多少次就归一了,暴力就好了(应该只有log(n)次) 于是暴力啊暴力,结果我归天了. 好吧,在各种题解的摧残下,我终于出了一篇巨好看(chou lou)代码(很多结构体党 ...

  4. loj#6029. 「雅礼集训 2017 Day1」市场(线段树)

    题意 链接 Sol 势能分析. 除法是不能打标记的,所以只能暴力递归.这里我们加一个剪枝:如果区间内最大最小值的改变量都相同的话,就变成区间减. 这样复杂度是\((n + mlogn) logV\)的 ...

  5. #6029. 「雅礼集训 2017 Day1」市场 [线段树]

    考虑到每次除法,然后加法,差距会变小,于是维护加法lazytag即可 #include <cstdio> #include <cmath> #define int long l ...

  6. 「长乐集训 2017 Day10」划分序列 (二分 dp)

    「长乐集训 2017 Day10」划分序列 题目描述 给定一个长度为 n nn 的序列 Ai A_iA​i​​,现在要求把这个序列分成恰好 K KK 段,(每一段是一个连续子序列,且每个元素恰好属于一 ...

  7. loj6271 「长乐集训 2017 Day10」生成树求和 加强版(矩阵树定理,循环卷积)

    loj6271 「长乐集训 2017 Day10」生成树求和 加强版(矩阵树定理,循环卷积) loj 题解时间 首先想到先分开三进制下每一位,然后每一位分别求结果为0,1,2的树的个数. 然后考虑矩阵 ...

  8. loj #6032. 「雅礼集训 2017 Day2」水箱 线段树优化DP转移

    $ \color{#0066ff}{ 题目描述 }$ 给出一个长度为 \(n\) 宽度为 \(1\) ,高度无限的水箱,有 \(n-1\) 个挡板将其分为 \(n\) 个 \(1 - 1\) 的小格, ...

  9. loj6271「长乐集训 2017 Day10」生成树求和 加强版

    又是一个矩阵树套多项式的好题. 这里我们可以对每一位单独做矩阵树,但是矩阵树求的是边权积的和,而这里我们是要求加法,于是我们i将加法转化为多项式的乘法,其实这里相当于一个生成函数?之后如果我们暴力做的 ...

随机推荐

  1. Linux系统资源查询命令(cpu、io、mem)

    cpu/mem: 1. 指定pid top -p pid1,pid2,... 2. top排序 先top,然后  输入大写P,则结果按CPU占用降序排序.输入大写M,结果按内存占用降序排序. io: ...

  2. Linux基本命令 权限管理命令

    1.权限管理命令chmod ================================================================================== 命令名 ...

  3. gstreamer——文档/资源/使用

    http://gstreamer.freedesktop.org/src/ http://gstreamer.freedesktop.org/data/doc/gstreamer/head/qt-gs ...

  4. window7 3G/4G拨号操作

    Win7系统Modem拨号操作指导:https://wenku.baidu.com/view/bb855b1dc77da26925c5b0e1.html 拨号上网设置APN,拨号号码,帐号和密码:ht ...

  5. no xxx find in java.library.path

    JAVA系统运行时候load native lib时候会遇到下面错误,如 java.lang.UnsatisfiedLinkError: no JSTAF in java.library.path这可 ...

  6. 使用ASP.Net MVC5 Web API OData和Sencha Touch 开发WebAPP

    使用ASP.Net MVC5 Web API OData和SenCha Touch 开发WebAPP Demo 效果 第一步 创建数据库 创建表 第二步 搭建MVC,并导入OData 第三步,写入We ...

  7. 3.6《深入理解计算机系统》笔记(四)虚拟存储器,malloc,垃圾回收【插图】

    概述 ●我们电脑上运行的程序都是使用虚拟存储,跟物理内存根本不搭边. ●既然虚拟内存是在磁盘上的,为什么它又运行这么好,并没有感觉卡顿?这要感谢程序的局部性! ●虚拟存储器的调度是一个操作系统必须做好 ...

  8. char,uchar,0xff

    如果:char test = 0xFF: 此时:test != 0xFF://因为test为char类型,0xFF为int,所以编译器会将test转为int(-1),所以不等于 如果:uchar te ...

  9. NorFlash、NandFlash、eMMC比较区别【转】

    本文转载自:http://www.veryarm.com/1200.html 快闪存储器(英语:Flash Memory),是一种电子式可清除程序化只读存储器的形式,允许在操作中被多次擦或写的存储器. ...

  10. 设置 IntelliJ IDEA 主题和字体的方法

    1 前言 在博文「IntelliJ IDEA 之 HelloWorld 项目创建及相关配置文件介绍」中,我们已经用 IntelliJ IDEA 创建了第一个 Java 项目 HelloWorld,如下 ...