线段树练习 区间合并

题目链接http://www.codeforces.com/problemset/problem/145/E

蛮好的题目

一连串由4、7组成的字符串

两种操作:count 输出整个区间内最长的不下降子序列的长度

switch a b 将a、b之间的数4变成7   7变成4

注意,最长不下贱子序列不一定要连续

一般情况下,不连续的最长不下降子序列的长度是无法维护的,但是这道题目的特殊性显而易见,只有4 7 两种数字,相当于0 1 

做法:

1:分别记录区间最长不下降和最长不上升的序列的长度,这样子进行异或操作的时候可以直接互换

2:记录区间0的个数和1的个数,利用这两个信息可以   在将信息往上传的时候  维护区间最长的不下降和不上升序列的长度,这里仔细想想

View Code
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int maxn = 1000010;
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
char str[maxn];
struct node{
int col;
int n0,n1,sum,ld;
void make(){
col^=1;
n0^=n1;n1=n0^n1;n0^=n1;
sum^=ld;ld=sum^ld;sum^=ld;
}
void init(){
col=n0=n1=sum=ld=0;
}
}a[maxn<<2];
int max(int a,int b){
return a>b?a:b;
}
void pushdown(int rt){
if(a[rt].col){
a[rt].col=0;
a[rt<<1].make();
a[rt<<1|1].make();
}
}
void pushup(int rt){
a[rt].n0=a[rt<<1].n0+a[rt<<1|1].n0;
a[rt].n1=a[rt<<1].n1+a[rt<<1|1].n1;
a[rt].sum=max(a[rt<<1].sum+a[rt<<1|1].n1,a[rt<<1|1].sum+a[rt<<1].n0);
a[rt].ld=max(a[rt<<1].ld+a[rt<<1|1].n0,a[rt<<1|1].ld+a[rt<<1].n1);
}
void update(int L,int R,int l,int r,int rt){
if(L<=l&&r<=R){
a[rt].make();
return ;
}
pushdown(rt);
int m=(l+r)>>1;
if(L<=m) update(L,R,lson);
if(R>m) update(L,R,rson);
pushup(rt);
}
void build(int l,int r,int rt){
a[rt].init();
if(l==r){
a[rt].n1=(str[l]=='7');
a[rt].n0=!a[rt].n1;
a[rt].ld=a[rt].sum=1;
return ;
}
int m=(l+r)>>1;
build(lson);
build(rson);
pushup(rt);
}
int main(){
int n,m,l,r;
char op[10];
scanf("%d%d",&n,&m);
scanf("%s",str+1);
build(1,n,1);
while(m--){
scanf("%s",op);
if(op[0]=='s') {
scanf("%d%d",&l,&r);
update(l,r,1,n,1);
}
else printf("%d\n",a[1].sum);
}
}



posted @ 2012-04-03 18:45  Because Of You  Views(448)  Comments(0Edit  收藏  举报