Time Limit: 3000 ms   Memory Limit: 256 MB

Description

  给你一个无限长的数组,初始的时候都为0,有3种操作:

  操作1是把给定区间$[l,r]$设为1,

  操作2是把给定区间$[l,r]$设为0,

  操作3把给定区间$[l,r]$0,1反转。

  一共n个操作,每次操作后要输出最小位置的0。

Input

  第一行一个整数n,表示有n个操作

  接下来n行,每行3个整数op,l,r表示一个操作

Output

  共n行,一行一个整数表示答案

Sample Input

Sample Output

3
1 3 4
3 1 6
2 1 3
1
3
1

HINT

  对于30%的数据$1\le n\le 10^3,1\le l\le r\le 10^{18}$

  对于100%的数据$1\le n\le 10^5,1\le l\le r\le 10^{18}$


题解

  离散化操作区间:

  首先看到$1\le l\le r\le 10^{18}$的范围,第一反应离散化。

    这题的离散化是非常讲究的,我们不能只把每个区间操作的两端点提取出来离散化(我就是这么干的,其他部分都是对的,结果眼睁睁地没法输出),因为这样无法考虑到区间之间的点,如下图:

  

  考虑到答案的位置,答案应该只可能出现在某一段操作区间右端点的右边一位,于是我们在离散一个操作区间$l,r$的同时,把$r+1$也离散化掉。操作并不涉及到$r+1$,仅仅是为了输出答案的正确和可行性。

  实现修改操作:

    看到熟悉的区间操作,当然要想想线段树啦。

     这里采用两棵线段树的写法实现3种操作,当然也有一棵线段树搞定的写法。

     两棵线段树$A,B$,都先按照离散化的规模建好,以离散化端点编号为索引。$A$记录$0$的信息,$B$记录$1$的信息。

     线段树维护的信息是:$0$或$1$最左出现的位置。

     第1个操作:将$A$的相应区间的信息清空(设置成最大值,因为不存在$0$了),将$B$的相应区间的信息填充(设置成区间的左端点位置),并打上清空或填充标记。日后记得下传。

     第2个操作:与第1个操作完全相反。

       第3个操作:将$A$的相应区间节点和$B$的相应区间节点对调。

  输出:

    询问$A$中的最小值位置,输出离散化前的原值即可。可以发现这个点一定是某个区间的$r$再$+1$。

  时间复杂度$O(n lg n)$,空间复杂度$O(n)$。


 #include <cstdio>
#include <algorithm>
#define min(a,b) (a<b?a:b)
using namespace std;
typedef long long ll;
const int N=1e5+;
ll INF=;
int n,lshtot,opt[N][],total;
ll inp[N][],lis[N*],minloc,maxloc,orival[N*];
void lshAndfill(){
sort(lis+,lis++lshtot);
total=unique(lis+,lis++lshtot)-lis-;
for(int i=;i<=total;i++) orival[i]=lis[i];
orival[total+]=INF;
for(int i=;i<=n;i++){
opt[i][]=inp[i][];
opt[i][]=lower_bound(lis+,lis++total,inp[i][])-lis;
opt[i][]=lower_bound(lis+,lis++total,inp[i][])-lis;
}
}
struct Seg{
int cnt,root[],sz,ch[N*][],mark[N*];
ll info[N*];
void build(int Size){
sz=Size;
_build(root[],,sz,true);
_build(root[],,sz,false);
}
void _build(int &u,int l,int r,bool isfill){
if(!u) u=++cnt;
mark[u]=-;
if(l==r){
if(isfill) info[u]=l;
else info[u]=total+;
return;
}
int mid=(l+r)>>;
_build(ch[u][],l,mid,isfill);
_build(ch[u][],mid+,r,isfill);
pushup(u);
}
inline void pushup(int u){
info[u]=min(info[ch[u][]],info[ch[u][]]);
}
inline void pushdown(int u,int l,int r){
int lc=ch[u][],rc=ch[u][];
if(mark[u]==-) return;
mark[lc]=mark[rc]=mark[u];
if(mark[u]==)
info[lc]=info[rc]=total+;
else{
info[lc]=l;
info[rc]=(l+r)/+;
}
mark[u]=-;
}
void setSeg(int flag,int l,int r){
_setSeg(root[^flag],root[^flag],,sz,l,r);
}
void _setSeg(int u1,int u2,int l,int r,int L,int R){
if(L<=l&&r<=R){
info[u1]=total+;
info[u2]=l;
mark[u1]=; mark[u2]=;
return;
}
pushdown(u1,l,r);
pushdown(u2,l,r);
int mid=(l+r)>>;
if(L<=mid) _setSeg(ch[u1][],ch[u2][],l,mid,L,R);
if(mid<R) _setSeg(ch[u1][],ch[u2][],mid+,r,L,R);
pushup(u1);
pushup(u2);
}
void swapSeg(int l,int r){_swapSeg(root[],root[],,sz,l,r);}
void _swapSeg(int &u1,int &u2,int l,int r,int L,int R){
if(L<=l&&r<=R){
swap(u1,u2);
return;
}
pushdown(u1,l,r);
pushdown(u2,l,r);
int mid=(l+r)>>;
if(L<=mid) _swapSeg(ch[u1][],ch[u2][],l,mid,L,R);
if(mid<R) _swapSeg(ch[u1][],ch[u2][],mid+,r,L,R);
pushup(u1);
pushup(u2);
}
inline ll getMin(int x){return info[root[x]];}
}seg;
int main(){
scanf("%d",&n);
minloc=-;
for(int i=;i<=n;i++){
scanf("%lld%lld%lld",&inp[i][],&inp[i][],&inp[i][]);
lis[++lshtot]=inp[i][];
lis[++lshtot]=inp[i][];
lis[++lshtot]=inp[i][]+;
if(minloc==-) minloc=min(inp[i][],inp[i][]);
else minloc=min(minloc,min(inp[i][],inp[i][]));
}
lshAndfill();
seg.build(total);
for(int i=;i<=n;i++){
if(minloc>){
printf("1\n");
continue;
}
if(opt[i][]<=)
seg.setSeg(opt[i][]==,opt[i][],opt[i][]);
else
seg.swapSeg(opt[i][],opt[i][]);
printf("%lld\n",orival[seg.getMin()]);
}
return ;
}

奇妙代码

mex (离散化+线段树)的更多相关文章

  1. 【XSY2484】mex 离散化 线段树

    题目大意 给你一个无限长的数组,初始的时候都为\(0\),有3种操作: 操作\(1\)是把给定区间\([l,r]\)设为\(1\): 操作\(2\)是把给定区间\([l,r]\)设为\(0\): 操作 ...

  2. 南阳理工 题目9:posters(离散化+线段树)

    posters 时间限制:1000 ms  |  内存限制:65535 KB 难度:6   描述 The citizens of Bytetown, AB, could not stand that ...

  3. SGU 180 Inversions(离散化 + 线段树求逆序对)

    题目链接:http://acm.sgu.ru/problem.php?contest=0&problem=180 解题报告:一个裸的求逆序对的题,离散化+线段树,也可以用离散化+树状数组.因为 ...

  4. 【POJ】2528 Mayor's posters ——离散化+线段树

    Mayor's posters Time Limit: 1000MS    Memory Limit: 65536K   Description The citizens of Bytetown, A ...

  5. hpu校赛--雪人的高度(离散化线段树)

    1721: 感恩节KK专场——雪人的高度 时间限制: 1 Sec  内存限制: 128 MB 提交: 81  解决: 35 [提交][状态][讨论版] 题目描述 大雪过后,KK决定在春秋大道的某些区间 ...

  6. 【BZOJ1645】[Usaco2007 Open]City Horizon 城市地平线 离散化+线段树

    [BZOJ1645][Usaco2007 Open]City Horizon 城市地平线 Description Farmer John has taken his cows on a trip to ...

  7. 【bzoj4636】蒟蒻的数列 离散化+线段树

    原文地址:http://www.cnblogs.com/GXZlegend/p/6801379.html 题目描述 蒟蒻DCrusher不仅喜欢玩扑克,还喜欢研究数列 题目描述 DCrusher有一个 ...

  8. 离散化+线段树/二分查找/尺取法 HDOJ 4325 Flowers

    题目传送门 题意:给出一些花开花落的时间,问某个时间花开的有几朵 分析:这题有好几种做法,正解应该是离散化坐标后用线段树成端更新和单点询问.还有排序后二分查找询问点之前总花开数和总花凋谢数,作差是当前 ...

  9. Mayor's posters (离散化线段树+对lazy的理解)

    题目 题意: n(n<=10000) 个人依次贴海报,给出每张海报所贴的范围 li,ri(1<=li<=ri<=10000000) .求出最后还能看见多少张海报. 思路: 由于 ...

  10. 干物妹小埋 (离散化 + 线段树 + DP)

    链接:https://ac.nowcoder.com/acm/contest/992/B来源:牛客网 题目描述 在之前很火的一个动漫<干物妹小埋>中,大家对小埋打游戏喝可乐的印象十分的深刻 ...

随机推荐

  1. Java进阶篇(六)——Swing程序设计(下)

    三.布局管理器 Swing中,每个组件在容器中都有一个具体的位置和大小,在容器中摆放各自组件时很难判断其具体位置和大小,这里我们就要引入布局管理器了,它提供了基本的布局功能,可以有效的处理整个窗体的布 ...

  2. javabean内省

    何为JavaBean? JavaBean 是一种JAVA语言写成的可重用组件.为写成JavaBean,类必须是具体的和公共的,并且具有无参数的构造器.JavaBean 通过提供符合一致性设计模式的公共 ...

  3. MySQL中group_concat()函数的排序方法

    group_concat()函数的参数是可以直接使用order by排序的.666..下面通过例子来说明,首先看下面的t1表. 比如,我们要查看每个人的多个分数,将该人对应的多个分数显示在一起,分数要 ...

  4. web.xml 中CharacterEncodingFilter类的学习

    过滤器配置 当前台JSP页面和JAVA代码中使用了不同的字符集进行编码的时候就会出现表单提交的数据或者上传/下载中文名称文件出现乱码的问题 //编码方式配置 <filter> <fi ...

  5. jQuery图片延迟加载插件jquery.lazyload.js

    在实际的项目开发中,我们通常会遇见这样的场景:一个页面有很多图片,而首屏出现的图片大概就一两张,那么我们还要一次性把所有图片都加载出来吗?显然这是愚蠢的,不仅影响页面渲染速度,还浪费带宽.这也就是们通 ...

  6. VUE-脚手架搭建

    1.什么vue-cli    vue-cli是vue.js的脚手架,用于自动生成vue.js工程模板的. 步骤: 2.安装   ->全局安装   npm install vue-cli -g 或 ...

  7. MVVM之旅(1)创建一个最简单的MVVM程序

    这是MVVM之旅系列文章的第一篇,许多文章和书喜欢在开篇介绍某种技术的诞生背景和意义,但是我觉得对于程序员来说,一个能直接运行起来的程序或许能够更直观的让他们了解这种技术.在这篇文章里,我将带领大家一 ...

  8. linux下的打包与压缩

    linux压缩或解压缩工具有很多,除了已经很少有人使用的compress外,现在常用的还有tar,bzip2,xz 和gziplinux压缩或解压缩工具有很多,除了已经很少有人使用的compress外 ...

  9. linux上的用户管理

    作为一个专业的服务器发行版,CentOS上存在着n多个用户,作为一个专业的运维工程师,linux的用户管理和用户组的管理是做为一个SA必不可少的工作. 下面我们就先来谈谈linux中的用户管理: us ...

  10. vsftp实现只能上传不能下载、删除权限配置

    vsftpd可以对每个用户特别限制.只要给那个用户建立一个设置文件,然后在文件里设置 在vsftpd.conf里加user_config_dir=/etc/vsftpd/vsftpd_user_con ...