bzoj 1858. [Scoi2010]序列操作 题解
Description
题目链接:BZOJ1858 [Scoi2010]序列操作
给你长度为n的01串,现有m个操作。
- 区间赋值为0
- 区间赋值为1
- 区间反转(所有0变为1,所有1变为0)
- 区间求和
- 求区间最多有连续的几个1
对于100%的数据,1le n,m le 100000。
Solution
珂朵莉树,如果您还不了解珂朵莉树,可以参见我的这篇博客。
由于Luogu上的数据已加强,故只能过3个点,但bzoj可过。
Code
代码语言:javascript复制#include<bits/stdc .h>
#define I inline
#define It set<node>::iterator
using namespace std;
inline int read(){int res=0,f=1;char ch=getchar();while(ch<'0'ch>'9'){if(ch=='-') f=-1;ch=getchar();}while(ch>='0'&&ch<='9') res=res*10 ch-'0',ch=getchar();return res*f;}
inline void write(int x){if(x<0) putchar('-'),x=-x;if(x<10) putchar(x '0');else write(x/10),putchar(x '0');}
struct node{
int l,r;mutable int v;
node(int L,int R=-1,int V=0):l(L),r(R),v(V){}
bool operator<(const node& x)const{return l<x.l;}
};
set<node> s;
I It split(int x){
It it=s.lower_bound(x);
if(it!=s.end()&&it->l==x) return it;
it--;
int L=it->l,R=it->r,V=it->v;
return s.erase(it),s.insert(node(L,x-1,V)),s.insert(node(x,R,V)).first;
}
I void assign(int l,int r,int v){
It it2=split(r 1),it1=split(l);
s.erase(it1,it2);s.insert(node(l,r,v));
}
I void flip(int l,int r){
It it2=split(r 1),it1=split(l);
for(;it1!=it2;it1 ) it1->v^=1;
}
I int sum(int l,int r){
It it2=split(r 1),it1=split(l);int s=0;
for(;it1!=it2;it1 ) s =it1->v*(it1->r-it1->l 1);
return s;
}
I int getmax(int l,int r){
It it2=split(r 1),it1=split(l);int s=0,Max=0;
for(;it1!=it2;it1 )
if(it1->v==1) s =it1->r-it1->l 1,Max=max(Max,s);
else s=0;
return max(Max,s);
}
int n,m;
int main(){
n=read(),m=read();
for(int x,i=0;i<n;i ) x=read(),s.insert(node(i,i,x));
s.insert(node(n,n,0));
for(int op,l,r,i=1;i<=m;i ){
op=read(),l=read(),r=read();
if(op==0) assign(l,r,0);
if(op==1) assign(l,r,1);
if(op==2) flip(l,r);
if(op==3) write(sum(l,r)),putchar('n');
if(op==4) write(getmax(l,r)),putchar('n');
}
}