比较烦的线段树

首先询问3很弱智不说,

询问4以前做过类似的,好像是USACO月赛hotel那题类似,维护lmax,rmax,max三个域就可以了

操作0,操作1也很简单,仔细考虑一下就知道也是可以lazy tag的

重点是操作2,好像数据结构题中翻转总是一个难点

由于翻转影响对询问4影响较大(对询问3基本没什么影响)

我们考虑这次维护六个域lmax0,rmax0,max0,lmax1,rmax1 max1 分别表示在区间内0和1从左起最多连续几个,从右起最多连续几个,区间内最多连续几个

然后我们来考虑操作,首先难想到维护rev域表示区间是否翻转,lazy表示0表示全0,1表示全1,-1表示没有进行覆盖操作

操作0,1对操作2有强制性,只要执行操作0,1,就一定可以直接覆盖之前的操作

然后执行取反操作的时候,如果lazy<>-1 那么对lazy取反(0-->1 1-->0) 否则再对rev取反

 type node=record
       l,r,lm1,rm1,m1,lm0,rm0,m0,sum:longint;
     end; var tree:array[..] of node;
    lazy,a:array[..] of integer;
    rev:array[..] of boolean;
    ch,i,n,m,x,y:longint; function max(a,b:longint):longint;
  begin
    if a>b then exit(a) else exit(b);
  end; function min(a,b:longint):longint;
  begin
    if a>b then exit(b) else exit(a);
  end; procedure swap(var a,b:longint);
  var c:longint;
  begin
    c:=a;
    a:=b;
    b:=c;
  end; procedure fill1(i:longint);
  var l,r:longint;
  begin
    l:=tree[i].l;
    r:=tree[i].r;
    tree[i].sum:=r-l+;
    tree[i].lm1:=r-l+;
    tree[i].rm1:=r-l+;
    tree[i].m1:=r-l+;
    tree[i].lm0:=;
    tree[i].rm0:=;
    tree[i].m0:=;
  end; procedure fill0(i:longint);
  var l,r:longint;
  begin
    l:=tree[i].l;
    r:=tree[i].r;
    tree[i].sum:=;
    tree[i].lm0:=r-l+;
    tree[i].rm0:=r-l+;
    tree[i].m0:=r-l+;
    tree[i].lm1:=;
    tree[i].rm1:=;
    tree[i].m1:=;
  end; procedure change(i:longint);
  var l,r:longint;
  begin
    l:=tree[i].l;
    r:=tree[i].r;
    tree[i].sum:=(r-l+)-tree[i].sum;
    swap(tree[i].m1,tree[i].m0);
    swap(tree[i].lm1,tree[i].lm0);
    swap(tree[i].rm1,tree[i].rm0);
  end; procedure update(i:longint);   //由下向上更新
  var l,r,m,p:longint;
  begin
    l:=tree[i].l;
    r:=tree[i].r;
    m:=(l+r) shr ;
    tree[i].sum:=tree[i*].sum+tree[i*+].sum;
//
    tree[i].lm1:=tree[i*].lm1;
    if tree[i*].lm1=m-l+ then
      tree[i].lm1:=tree[i].lm1+tree[i*+].lm1;     tree[i].rm1:=tree[i*+].rm1;
    if tree[i*+].rm1=r-m then
      tree[i].rm1:=tree[i].rm1+tree[i*].rm1;     tree[i].m1:=max(tree[i].lm1,tree[i].rm1);
    p:=max(tree[i*].rm1+tree[i*+].lm1,max(tree[i*].m1,tree[i*+].m1));
    tree[i].m1:=max(p,tree[i].m1);
//
    tree[i].lm0:=tree[i*].lm0;
    if tree[i*].lm0=m-l+ then
      tree[i].lm0:=tree[i].lm0+tree[i*+].lm0;     tree[i].rm0:=tree[i*+].rm0;
    if tree[i*+].rm0=r-m then
      tree[i].rm0:=tree[i].rm0+tree[i*].rm0;     tree[i].m0:=max(tree[i].lm0,tree[i].rm0);
    p:=max(tree[i*].rm0+tree[i*+].lm0,max(tree[i*].m0,tree[i*+].m0));
    tree[i].m0:=max(p,tree[i].m0);
  end; procedure pushdown(i:longint);   //传递标记
  begin
    if lazy[i]<>- then
    begin
      lazy[i*]:=lazy[i];
      lazy[i*+]:=lazy[i];
      if lazy[i]= then
      begin
        fill1(i*);
        fill1(i*+);
      end
      else begin
        fill0(i*);
        fill0(i*+);
      end;
      lazy[i]:=-;
      rev[i]:=false;  
    end;
    if rev[i] then
    begin
      if lazy[i*]<>- then lazy[i*]:=-lazy[i*]
      else rev[i*]:=not rev[i*];
      if lazy[i*+]<>- then lazy[i*+]:=-lazy[i*+]
      else rev[i*+]:=not rev[i*+];
      change(i*+);
      change(i*);
      rev[i]:=false;
    end;
  end; procedure build(i,l,r:longint);
  var m:longint;
  begin
    tree[i].l:=l;
    tree[i].r:=r;
    lazy[i]:=-;
    if l=r then
    begin
      tree[i].sum:=a[l];
      if a[l]= then
      begin
        tree[i].lm1:=;
        tree[i].rm1:=;
        tree[i].m1:=;
      end
      else begin
        tree[i].lm0:=;
        tree[i].rm0:=;
        tree[i].m0:=;
      end;
    end
    else begin
      m:=(l+r) shr ;
      build(i*,l,m);
      build(i*+,m+,r);
      update(i);
    end;
  end; procedure work(i,l,r:longint);
  var m:longint;
  begin
    if (x<=l) and (y>=r) then
    begin
      if rev[i]=true then rev[i]:=false;
      lazy[i]:=ch;
      if ch= then fill1(i)
      else fill0(i);
    end
    else begin
      if (lazy[i]<>-) or (rev[i]) then pushdown(i);
      m:=(l+r) shr ;
      if x<=m then work(i*,l,m);
      if y>=m+ then work(i*+,m+,r);
      update(i);
    end;
  end; procedure reverse(i,l,r:longint);
  var m:longint;
  begin
    if (x<=l) and (y>=r) then
    begin
      if lazy[i]<>- then lazy[i]:=-lazy[i]
      else rev[i]:=not rev[i];
      change(i);
    end
    else begin
      if (lazy[i]<>-) or rev[i] then pushdown(i);
      m:=(l+r) shr ;
      if x<=m then reverse(i*,l,m);
      if y>=m+ then reverse(i*+,m+,r);
      update(i);
    end;
  end; function asksum(i,l,r:longint):longint;
  var m,t:longint;
  begin
    if (x<=l) and (y>=r) then exit(tree[i].sum)
    else begin
      if (lazy[i]<>-) or rev[i] then pushdown(i);
      m:=(l+r) shr ;
      t:=;
      if (x<=m) then t:=t+asksum(i*,l,m);
      if y>=m+ then t:=t+asksum(i*+,m+,r);
      update(i);
      exit(t);
    end;
  end; function askmax(i,l,r:longint):longint;
  var m,t,t1,t2:longint;
  begin
    if (x<=l) and (y>=r) then exit(tree[i].m1)
    else begin
      if (lazy[i]<>-) or rev[i] then pushdown(i);
      m:=(l+r) shr ;
      t1:=;
      t2:=;
      t:=;
      if (x<=m) then t1:=askmax(i*,l,m);
      if y>=m+ then t2:=askmax(i*+,m+,r);
      if (x<=m) and (y>=m+) then t:=min(tree[i*].rm1,m-x+)+min(tree[i*+].lm1,y-m); //注意不能少考虑这种情况
      t:=max(t,max(t1,t2));
      update(i);
      exit(t);
    end;
  end; begin
  readln(n,m);
  for i:= to n do
    read(a[i]);
  build(,,n);   //初始化
  for i:= to m do
  begin
    readln(ch,x,y);
    inc(x);
    inc(y);
    if ch= then
      work(,,n)
    else if ch= then
      work(,,n)
    else if ch= then
      reverse(,,n)
    else if ch= then
      writeln(asksum(,,n))
    else if ch= then
      writeln(askmax(,,n));
  end;
end.

bzoj1858的更多相关文章

  1. 【BZOJ1858】序列操作(线段树)

    [BZOJ1858]序列操作(线段树) 题面 BZOJ 题解 这题思路很简单,细节很烦,很码 维护区间翻转和区间赋值标记 当打到区间赋值标记时直接覆盖掉翻转标记 下放标记的时候先放赋值标记再放翻转标记 ...

  2. scoi2010&&bzoj1858序列操作

    [题目描述] lxhgww最近收到了一个01序列,序列里面包含了n个数,这些数要么是0,要么是1,现在对于这个序列有五种变换操作和询问操作: 0 a b 把[a, b]区间内的所有数全变成0 1 a ...

  3. 【BZOJ-1858】序列操作 线段树

    1858: [Scoi2010]序列操作 Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 1961  Solved: 991[Submit][Status ...

  4. BZOJ1858[Scoi2010]序列操作 题解

    题目大意: 有一个01序列,现在对于这个序列有五种变换操作和询问操作: 0 a b 把[a, b]区间内的所有数全变成0:1 a b 把[a, b]区间内的所有数全变成1:2 a b 把[a,b]区间 ...

  5. bzoj1858: [Scoi2010]序列操作

    lazy-tag线段树. #include<cstdio> #include<algorithm> #include<cstring> using namespac ...

  6. bzoj千题计划177:bzoj1858: [Scoi2010]序列操作

    http://www.lydsy.com/JudgeOnline/problem.php?id=1858 2018 自己写的第1题,一遍过 ^_^ 元旦快乐 #include<cstdio> ...

  7. bzoj1858[Scoi2010]序列操作 线段树

    1858: [Scoi2010]序列操作 Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 3079  Solved: 1475[Submit][Statu ...

  8. [待码][BZOJ1858]SCOI2010序列操作 jzyzoj1655

    待码的线段树.....太长了看上去不是很想写 [ 什么破理由啊摔,不要脸 ] 嗯先水几道再写

  9. 【bzoj1858】[Scoi2010]序列操作 线段树区间合并

    题目描述 lxhgww最近收到了一个01序列,序列里面包含了n个数,这些数要么是0,要么是1,现在对于这个序列有五种变换操作和询问操作: 0 a b 把[a, b]区间内的所有数全变成0 1 a b ...

随机推荐

  1. 浅析ASP.NET的状态保持

    ASP.NET的状态保持:1.viewstate:隐藏域,记录服务器端控件的状态,适用于页面不关闭的情况下多次与服务器交互,页面自己给自己传值:文本框的改变事件.IspostBack也依赖viewst ...

  2. Solr配置与简单Demo

    简介: solr是基于Lucene Java搜索库的企业级全文搜索引擎,目前是apache的一个项目.它的官方网址在http://lucene.apache.org/solr/  .solr需要运行在 ...

  3. struts2的单文件下载

    一. 导入两个jar包 commons-fileupload-1.3.1.jar commons-io-2.4.jar 二.编写请求上传jsp <h1>文件列表--单文件</h1&g ...

  4. .NET 设计模式之单例模式(一)

    1.简单解释:在创建对象时,无论创建多少次,在堆空间上只会申请一次内存空间. 2.例子(1): public class Singleton{ private static Singleton _si ...

  5. JSP EL表达式详细介绍

    一.JSP EL语言定义 E L(Expression Language)  目的:为了使JSP写起来更加简单. 表达式语言的灵感来自于 ECMAScript 和 XPath 表达式语言,它提供了在 ...

  6. Codevs 1507 酒厂选址

    1507 酒厂选址 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 黄金 Gold 传送门 题目描述 Description Abstinence(戒酒)岛的居民们酷爱一种无酒精啤酒 ...

  7. Cogs 1298.通讯问题

    1298.通讯问题 ★ 输入文件:jdltt.in 输出文件:jdltt.out 简单对比 时间限制:1 s 内存限制:128 MB [题目描述] 一个篮球队有n个篮球队员,每个队员都有联系方式(如电 ...

  8. 【实习记】2014-08-27堆排序理解总结+使用typedef指代函数指针

        过程记录 4个月前C语言版的七大排序算法实践让我在写C++版时轻车熟路.特别是冒泡,插入,希尔,选择这四种排序不用调试即运行成功.输出的效果与C语言做的版本完全一样,其中令我印象深刻的是,co ...

  9. 思维导图软件VYM

    http://www.insilmaril.de/vym/ 点击打开链接http://www.oschina.net/p/vym 有人说VYM就跟目录是一回事,确实是这样, 只不过与excel之类的比 ...

  10. PHP 插入排序法

    <?php function insertSort($arr) { //区分 哪部分是已经排序好的 //哪部分是没有排序的 //找到其中一个需要排序的元素 //这个元素 就是从第二个元素开始,到 ...