题目链接:

题目

Problem 1007 幸运数

Time Limit: 2000 mSec

Memory Limit : 131072 KB

问题描述

皮特的幸运数是2和5。只由幸运数字2和5组成的数列,称为幸运数列。对于幸运数列,有两种操作。

1、switch i j

表示数列从第i个数到第j个数中,所有的2改成5,所有的5改成2.例如幸运数列25525,执行switch 2 4操作,则数列变成22255

2、count

表示要求输出当前幸运数列的最长不下降子序列(即子序列中后面的数都不小于前面的数)的长度。例如幸运数列252255,其中222,555,2555是它的不下降子序列,而2525不是,可以看出,最长不下降子序列是22255,长度为5。

现在给出一个长度为n的幸运数列,再依次给出m个操作,对于每个count操作,输出当前幸运数列的最长不下降子序列的长度。

输入

输入包含多组数据。(数据不超过10组)

每组数据的第一行有两个整数n,m(n,m<=100000),分别表示数列的长度和操作个数。

接下来一行,有一个长度为n的只由2和5构成的数列。

接下来m行,每行一个操作,是count或者switch i j,含义如题所述

输出

对于每组数据的每个count操作,都输出一行,包含一个整数,表示当前幸运数列的最长不下降子序列的长度。

样例

input

2 3

25

count

switch 1 2

count

3 5

525

count

switch 1 1

count

switch 1 3

count

output

2

1

2

3

2

题解

可以用线段树跑成段更新。

每个节点维护四个变量,分别表示子序列"22","25","52","55"的长度

在2,5对换的时候,只要交换"22"和"55";"25"和"52"。

需要懒惰标记。(感觉懒惰标记写搓了。orz,后面学一下)

代码

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define lson (o<<1)
#define rson ((o<<1)|1)
#define M L+(R-L)/2
using namespace std; const int maxn=101010; int sumv[maxn<<2][4],setv[maxn<<2];
int n,m;
char str[maxn]; void pushdown(int o,int L,int R){
if(setv[o]>0){
swap(sumv[o][1],sumv[o][2]);
swap(sumv[o][0],sumv[o][3]);
if(L<R){
setv[lson]^=1; setv[rson]^=1;
}
setv[o]=0;
}
} void maintain(int o,int L,int R){
if(L==R){
pushdown(o,L,R);
}else{
pushdown(lson,L,M);
pushdown(rson,M+1,R);
sumv[o][0]=sumv[lson][0]+sumv[rson][0];
sumv[o][3]=sumv[lson][3]+sumv[rson][3];
sumv[o][1]=max(sumv[lson][0]+sumv[rson][3],sumv[lson][1]+sumv[rson][3]);
sumv[o][1]=max(sumv[o][1],sumv[lson][0]+sumv[rson][1]);
sumv[o][2]=max(sumv[lson][3]+sumv[rson][0],sumv[lson][2]+sumv[rson][0]);
sumv[o][2]=max(sumv[o][2],sumv[lson][3]+sumv[rson][2]);
}
} int ql,qr;
void update(int o,int L,int R){
if(ql<=L&&R<=qr){
setv[o]^=1;
pushdown(o,L,R);
}else{
pushdown(o,L,R);
if(ql<=M) update(lson,L,M);
if(qr>M) update(rson,M+1,R);
}
maintain(o,L,R);
} void build(int o,int L,int R){
if(L==R){
memset(sumv[o],0,sizeof(sumv[o]));
if(str[L]=='2') sumv[o][0]=1;
else sumv[o][3]=1;
}else{
build(lson,L,M);
build(rson,M+1,R);
}
maintain(o,L,R);
} void init(){
memset(setv,0,sizeof(setv));
} int main(){
while(scanf("%d%d",&n,&m)==2){
init();
scanf("%s",str+1);
build(1,1,n);
char cmd[22];
while(m--){
scanf("%s",cmd);
if(cmd[0]=='c'){
printf("%d\n",max(sumv[1][0],max(sumv[1][1],sumv[1][3])));
}else{
scanf("%d%d",&ql,&qr);
update(1,1,n);
}
}
}
return 0;
}

改进版本:(看起来爽多了)

打标记的位置其实已经更新完毕了,打标记知识告诉后代要更新啦。

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define lson (o<<1)
#define rson ((o<<1)|1)
#define M L+(R-L)/2
using namespace std; const int maxn = 101010; int sumv[maxn << 2][4], setv[maxn << 2];
int n, m;
char str[maxn]; void pushdown(int o, int L, int R) {
if (setv[o]>0&&L<R) {
swap(sumv[lson][1], sumv[lson][2]); swap(sumv[lson][0], sumv[lson][3]);
swap(sumv[rson][1], sumv[rson][2]); swap(sumv[rson][0], sumv[rson][3]);
setv[lson] ^= 1; setv[rson] ^= 1;
setv[o] = 0;
}
} void maintain(int o, int L, int R) {
sumv[o][0] = sumv[lson][0] + sumv[rson][0];
sumv[o][3] = sumv[lson][3] + sumv[rson][3];
sumv[o][1] = max(sumv[lson][0] + sumv[rson][3], sumv[lson][1] + sumv[rson][3]);
sumv[o][1] = max(sumv[o][1], sumv[lson][0] + sumv[rson][1]);
sumv[o][2] = max(sumv[lson][3] + sumv[rson][0], sumv[lson][2] + sumv[rson][0]);
sumv[o][2] = max(sumv[o][2], sumv[lson][3] + sumv[rson][2]);
} int ql, qr;
void update(int o, int L, int R) {
if (ql <= L&&R <= qr) {
setv[o] ^= 1;
swap(sumv[o][1], sumv[o][2]);
swap(sumv[o][0], sumv[o][3]);
}
else {
pushdown(o, L, R);
if (ql <= M) update(lson, L, M);
if (qr>M) update(rson, M + 1, R);
maintain(o, L, R);
}
} void build(int o, int L, int R) {
if (L == R) {
memset(sumv[o], 0, sizeof(sumv[o]));
if (str[L] == '2') sumv[o][0] = 1;
else sumv[o][3] = 1;
}
else {
build(lson, L, M);
build(rson, M + 1, R);
maintain(o, L, R);
}
} void init() {
memset(setv, 0, sizeof(setv));
} int main() {
while (scanf("%d%d", &n, &m) == 2) {
init();
scanf("%s", str + 1);
build(1, 1, n);
char cmd[22];
while (m--) {
scanf("%s", cmd);
if (cmd[0] == 'c') {
printf("%d\n", max(sumv[1][0], max(sumv[1][1], sumv[1][3])));
}
else {
scanf("%d%d", &ql, &qr);
update(1, 1, n);
}
}
}
return 0;
}

Problem 1007 幸运数 线段树成段更新的更多相关文章

  1. poj 3468 A Simple Problem with Integers 【线段树-成段更新】

    题目:id=3468" target="_blank">poj 3468 A Simple Problem with Integers 题意:给出n个数.两种操作 ...

  2. POJ 3468:A Simple Problem with Integers(线段树[成段更新])

    题意:N个数Q次操作.一共两种操作:Q l r :询问[l,r]这个区间里的数字和,C l r c: [l,r]区间里的每个数都加上c.1 ≤ N,Q ≤ 100000. 方法:线段树的成段更新.注意 ...

  3. poj 3468 A Simple Problem with Integers (线段树 成段更新 加值 求和)

    题目链接 题意: 只有这两种操作 C a b c" means adding c to each of Aa, Aa+1, ... , Ab. -10000 ≤ c ≤ 10000.&quo ...

  4. POJ3468_A Simple Problem with Integers(线段树/成段更新)

    解题报告 题意: 略 思路: 线段树成段更新,区间求和. #include <iostream> #include <cstring> #include <cstdio& ...

  5. 线段树(成段更新) POJ 3468 A Simple Problem with Integers

    题目传送门 /* 线段树-成段更新:裸题,成段增减,区间求和 注意:开long long:) */ #include <cstdio> #include <iostream> ...

  6. ACM: Copying Data 线段树-成段更新-解题报告

    Copying Data Time Limit:2000MS Memory Limit:262144KB 64bit IO Format:%I64d & %I64u Description W ...

  7. Codeforces Round #149 (Div. 2) E. XOR on Segment (线段树成段更新+二进制)

    题目链接:http://codeforces.com/problemset/problem/242/E 给你n个数,m个操作,操作1是查询l到r之间的和,操作2是将l到r之间的每个数xor与x. 这题 ...

  8. POJ 2777 Count Color (线段树成段更新+二进制思维)

    题目链接:http://poj.org/problem?id=2777 题意是有L个单位长的画板,T种颜色,O个操作.画板初始化为颜色1.操作C讲l到r单位之间的颜色变为c,操作P查询l到r单位之间的 ...

  9. HDU1698_Just a Hook(线段树/成段更新)

    解题报告 题意: 原本区间1到n都是1,区间成段改变成一个值,求最后区间1到n的和. 思路: 线段树成段更新,区间去和. #include <iostream> #include < ...

随机推荐

  1. 十二、Android UI开发专题(转)

    http://dev.10086.cn/cmdn/bbs/viewthread.php?tid=18736&page=1#pid89255Android UI开发专题(一) 之界面设计 近期很 ...

  2. 北大ACM(POJ1010-STAMPS)

    Question:http://poj.org/problem?id=1010问题点:DFS.剪枝. Memory: 220K Time: 32MS Language: C++ Result: Acc ...

  3. oracle删除字段时候判断字段是否存在

    declare v_count number; begin ) into v_count from all_tab_columns a where a.TABLE_NAME = 'XXX1' and ...

  4. [Android开发系列]IT博客应用

    1.关于坑 好吧,在此之前先来说一下,之前开的坑,恩,确实是坑,前面开的两个android开发教程的坑,对不起,实在是没什么动力了,不过源码都有的,大家可以参照github这个应用 https://g ...

  5. HTML+CSS学习笔记 (13) - CSS代码缩写,占用更少的带宽

    标签:HTML+CSS 盒模型代码简写 还记得在讲盒模型时外边距(margin).内边距(padding)和边框(border)设置上下左右四个方向的边距是按照顺时针方向设置的:上右下左.具体应用在m ...

  6. tcpServer 浅显的发一代码

    接下来发出来的一段代码也是我从网上找的一个例子,具体的来源已经找不到了,跟作者说声抱歉 ,现在公司做机票,出于性能的原因,就重写一个底层的tcp请求(不是我写的) 下面测试的是个控制台应用程序 Htt ...

  7. (转)Linux概念架构的理解

    英文原文:Conceptual Architecture of the Linux Kernel 摘要 Linux kernel成功的两个原因:(1)架构设计支持大量的志愿开发者加入到开发过程中:(2 ...

  8. 《锋利的jQuery》心得笔记--One Sections

    第一章 1.    $是jQuery的一个简写形式 2.    在jQuery中无法使用DOM对象的任何方法:比如:$ (“#id”).innerHTML.$ (“#id”).checked, 可以使 ...

  9. Hibernate中Entity实体类的写法

    记录下一个Entity类的写法,方便以后查阅: package com.bupt.auth.entity; import java.util.Date; import javax.persistenc ...

  10. 从基础知识到重写Spring的Bean工厂中学习java的工厂模式

    1.静态工厂模式其他对象不能直接通过new得到某个类,而是通过调用getInstance()方法得到该类的对象这样,就可以控制类的产生过程.顺带提一下单例模式和多例模式:  单例模式是指控制其他对象获 ...