bzoj 1858. [Scoi2010]序列操作 题解

2022-09-19 11:54:56 浏览数 (1)

bzoj 1858. [Scoi2010]序列操作 题解

Description

题目链接:BZOJ1858 [Scoi2010]序列操作

给你长度为n01串,现有m个操作。

  1. 区间赋值为0
  2. 区间赋值为1
  3. 区间反转(所有0变为1,所有1变为0
  4. 区间求和
  5. 求区间最多有连续的几个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');
}
}

0 人点赞