[Codeforces]817F. MEX Queries

You are given a set of integer numbers, initially it is empty. You should perform n queries. 
There are three different types of queries: 
1 l r — Add all missing numbers from the interval [l, r] 
2 l r — Remove all present numbers from the interval [l, r] 
3 l r — Invert the interval [l, r] — add all missing and remove all present numbers from the interval [l, r] 
After each query you should output MEX of the set — the smallest positive (MEX  ≥ 1) integer number which is not presented in the set. 
Input 
The first line contains one integer number n (1 ≤ n ≤ 105). 
Next n lines contain three integer numbers t, l, r (1 ≤ t ≤ 3, 1 ≤ l ≤ r ≤ 1018) — type of the query, left and right bounds. 
Output 
Print MEX of the set after each query. 
Examples 
input 

1 3 4 
3 1 6 
2 1 3 
output 



input 

1 1 3 
3 5 6 
2 4 4 
3 1 6 
output 




Note 
Here are contents of the set after each query in the first example: 
{3, 4} — the interval [3, 4] is added 
{1, 2, 5, 6} — numbers {3, 4} from the interval [1, 6] got deleted and all the others are added 
{5, 6} — numbers {1, 2} got deleted

题意

给你一个无限长的数组,初始的时候都为0,操作1是把给定区间清零,操作2是把给定区间设为1,操作3把给定区间反转。每次操作后要输出最小位置的0。

题解

看到数据范围n<=10^5,结合题意可以考虑使用线段树维护对区间的修改操作。但是l,r<=10^18,所以首先要离散化一下。在使用线段树维护的时候,节点维护该区间数相加的总和。对于操作1和操作2,我们分别赋值为1和0,对于操作3,我们把区间反转,那么新的区间和就是区间的长度减去原来的区间和。然后每次查询最小位置的0,只需要看一下左儿子所代表的区间是否小于这个区间的长度,如果是就在左儿子,否则就在右儿子查找。

题目细节

这道题有很多坑人的点,首先,在离散化的时候必须把1也加上,因为答案可能为1;线段树在下传标记时要注意顺序;记录原来信息的数组必须得开long long,空间一定要开够。

代码

 #include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<cmath>
#include<map>
using namespace std;
#define ll long long
#define REP(i,a,b) for(register int i=(a),_end_=(b);i<=_end_;i++)
#define DREP(i,a,b) for(register int i=(a),_end_=(b);i>=_end_;i--)
#define EREP(i,a) for(register int i=start[(a)];i;i=e[i].next)
inline int read()
{
int sum=,p=;char ch=getchar();
while(!((''<=ch && ch<='') || ch=='-'))ch=getchar();
if(ch=='-')p=-,ch=getchar();
while(''<=ch && ch<='')sum=sum*+ch-,ch=getchar();
return sum*p;
}
const int maxn=; map <ll,int> mp;
int m,cnt;
ll s[maxn*],n; struct qu {
ll l,r;
int type;
}a[maxn]; struct node {
int s,lz,id;//s记录区间和,lz为懒标记,id维护区间是否反转
}c[maxn*]; #define lc (o<<1)
#define rc (o<<1 | 1)
#define left lc,l,mid
#define right rc,mid+1,r inline void make_tree(int o,int l,int r)
{
c[o].s=;c[o].lz=-;c[o].id=;
if(l==r)return;
int mid=(l+r)>>;
make_tree(left);
make_tree(right);
} void maintain(int o,int l,int r)
{
c[o].s=c[lc].s+c[rc].s;
} void pushdown(int o,int l,int r)
{
int mid=(l+r)>>;
if(c[o].lz!=-)//下传懒标记,同时将儿子节点的反转标记清0
{
c[lc].lz=c[rc].lz=c[o].lz;
c[lc].s=(mid-l+)*c[o].lz;
c[rc].s=(r-mid)*c[o].lz;
c[lc].id=c[rc].id=;
c[o].lz=-;
}
if(c[o].id)//将儿子节点的反转标记也反转,同时维护儿子的区间和
{
c[lc].id^=;
c[rc].id^=;
c[lc].s=(mid-l+)-c[lc].s;
c[rc].s=(r-mid)-c[rc].s;
c[o].id=;
}
} inline void updates(int ql,int qr,int x,int o,int l,int r)
{
pushdown(o,l,r);
if(ql==l && r==qr)//把区间覆盖为x
{
c[o].s=(r-l+)*x;
c[o].lz=x;
c[o].id=;
return;
}
int mid=(l+r)>>;
if(ql>mid)
{
updates(ql,qr,x,right);
}
else if(qr<=mid)
{
updates(ql,qr,x,left);
}else
{
updates(ql,mid,x,left);
updates(mid+,qr,x,right);
}
maintain(o,l,r);
} inline void updatex(int ql,int qr,int o,int l,int r)
{
pushdown(o,l,r);
if(ql==l && r==qr)//把区间反转
{
c[o].s=(r-l+)-c[o].s;
c[o].id^=;
return;
}
int mid=(l+r)>>;
if(ql>mid)
{
updatex(ql,qr,right);
}
else if(qr<=mid)
{
updatex(ql,qr,left);
}else
{
updatex(ql,mid,left);
updatex(mid+,qr,right);
}
maintain(o,l,r);
} void init()
{
m=read();
REP(i,,m)
{
cin>>a[i].type>>a[i].l>>a[i].r;
a[i].r++;
s[++cnt]=a[i].l;
s[++cnt]=a[i].r;
}
s[++cnt]=;//答案中可能会有1,必须加上
sort(s+,s+cnt+);
n=unique(s+,s+cnt+)-(s+);
REP(i,,n)mp[s[i]]=i;
make_tree(,,n);
} void query(int o,int l,int r)
{
if(l==r)
{
cout<<s[l]<<endl;
return;
}
int mid=(l+r)>>;
pushdown(o,l,r);
if(c[lc].s<mid-l+)
query(left);
else query(right);
} void doing()
{
REP(i,,m)
{
int type=a[i].type,l=mp[a[i].l],r=mp[a[i].r]-;
if(type==)
{
updates(l,r,,,,n);
}
else if(type==)
{
updates(l,r,,,,n);
}else
{
updatex(l,r,,,n);
}
query(,,n);
}
} int main()
{
init();
doing();
return ;
}

[Codeforces]817F. MEX Queries 离散化+线段树维护的更多相关文章

  1. Educational Codeforces Round 23 F. MEX Queries 离散化+线段树

    F. MEX Queries time limit per test 2 seconds memory limit per test 256 megabytes input standard inpu ...

  2. codeforces Good bye 2016 E 线段树维护dp区间合并

    codeforces Good bye 2016 E 线段树维护dp区间合并 题目大意:给你一个字符串,范围为‘0’~'9',定义一个ugly的串,即串中的子串不能有2016,但是一定要有2017,问 ...

  3. Codeforces GYM 100114 D. Selection 线段树维护DP

    D. Selection Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/gym/100114 Descriptio ...

  4. Codeforces 817F MEX Queries

    题意:对一个维护三种操作:1.将[l..r]中的数全部加入集合中.2.将集合中[l..r]范围内的数删去.3.将集合中在[l..r]中的数删去,并将之前不在集合中的数加入集合 考虑到最近线段树总是写爆 ...

  5. mex (离散化+线段树)

    Time Limit: 3000 ms   Memory Limit: 256 MB Description 给你一个无限长的数组,初始的时候都为0,有3种操作: 操作1是把给定区间$[l,r]$设为 ...

  6. [Noi2016]区间[离散化+线段树维护+决策单调性]

    4653: [Noi2016]区间 Time Limit: 60 Sec  Memory Limit: 256 MBSubmit: 621  Solved: 329[Submit][Status][D ...

  7. SPOJ - GSS1-Can you answer these queries I 线段树维护区间连续和最大值

    SPOJ - GSS1:https://vjudge.net/problem/SPOJ-GSS1 参考:http://www.cnblogs.com/shanyr/p/5710152.html?utm ...

  8. 2019牛客多校第七场E Find the median 离散化+线段树维护区间段

    Find the median 题意 刚开始集合为空,有n次操作,每次操作往集合里面插入[L[i],R[i]]的值,问每次操作后中位数是多少 分析 由于n比较大,并且数可以达到1e9,我们无法通过权值 ...

  9. Codeforces 997E - Good Subsegments(线段树维护最小值个数+历史最小值个数之和)

    Portal 题意: 给出排列 \(p_1,p_2,p_3,\dots,p_n\),定义一个区间 \([l,r]\) 是好的当且仅当 \(p_l,p_{l+1},p_{l+2},\dots,p_r\) ...

随机推荐

  1. HttpServletRequest cannot be resolved to a type。

    问题描述:HttpServletRequest cannot be resolved to a type.Multiple markers at this line - The import java ...

  2. java中的nextLine

    package scanner; import java.util.Scanner; public class NextLine { public static void main(String[] ...

  3. 批处理注册dll时候 遇到错误:模块已加载,但对***dll的调用失败

    解决方法 在批处理的第一行加入:cd /d %~dp0 然后在批处理上右键选择使用管理员权限运行

  4. junit测试套件

    在实际项目中,随着项目进度的开展,单元测试类会越来越多,可是直到现在我们还只会一个一个的单独运行测试类,这在实际项目实践中肯定是不可行的.为了解决这个问题,JUnit 提供了一种批量运行测试类的方法, ...

  5. Web应用基础

    B-S架构 架构的发展 1,c/s架构 (client客户端-server服务端)         (胖客户端:要求客户端运行业务:把业务放到服务器端,则是瘦客户端)         典型的c/s应用 ...

  6. 李忠益TP5商城项目笔记(待完成)

    商品种类的无限极分类 $data=db('goods_type')->field(['*','concat(path,",",id)'=>'paths'])->o ...

  7. 【转】python入门指引

    http://matrix.42qu.com/10757179 前言 其实我也不知道python怎么入门,由我来写这个真的不是很合适.我学python是直接找了dive into python来看.然 ...

  8. 【转】十分有用的linux shell学习总结

    在最近的日常工作中由于经常会和Linux服务器打交道,如Oracle性能优化.我们 数据采集服务器的资源利用率监控,以及Debug服务器代码并解决其效率和稳定性等问题.因此这段时间总结的有关Linux ...

  9. 【转】awk用法介绍

    1. 介绍 Awk是由Aho.Weinberger和Kernighan三位科学家开发的,特别擅长文本处理的linux 工具,该工具是 linux 下最常用的工具之一.Awk不是普通的工具,其实,也是一 ...

  10. 高通ASOC中的machine驱动

    ASoC被分为Machine.Platform和Codec三大部分,其中的Machine驱动负责Platform和Codec之间的耦合以及部分和设备或板子特定的代码,再次引用上一节的内容:Machin ...