其实这两题都是基础的线段树,但对于我这个线段树的初学者来说,总结一下还是很有用的;

poj3468显然是线段树区间求和,区间更改的问题,而poj2528是对区间染色,问有多少种颜色的问题;

线段树的建立和求和附代码,还是比较简单的;

这里想说的是区间修改,用到了了lazy思想:打标记;

拿poj2528举例,比如对区间[l,r]染色,我们只要在线段树中,被[l,r]覆盖的最大子区间[p,q]上标记被染成了什么颜色即可,不需要再往下遍历[p,q]的左右孩子;当下次修改影响到了区间[p,q]时(区间有交集),说明[p,q]一定不会全都维持原来的颜色。我们将标记向下传递给左右孩子(同时自身标记清除),不断传递下去,直至某个区间完全被要修改区间覆盖,,再给这个区间打上新的标记。这样可保证时间复杂度为O(logn);

总之lazy思想的精髓就是,能不往下访问就不访问,要更改的时候再将子节点更改,从而减少时间复杂度;

 var lazy,tree:array[..] of int64;
l,n,m,j,x,a,b,i:longint;
ans:int64;
c:char;
procedure pushdown(l,r,i:longint);
var m:longint;
begin
m:=(l+r) div ;
if lazy[i]= then exit;
lazy[i*]:=lazy[i*]+lazy[i];
lazy[i*+]:=lazy[i*+]+lazy[i];
tree[i*]:=tree[i*]+lazy[i]*(m-l+);
tree[i*+]:=tree[i*+]+lazy[i]*(r-m);
lazy[i]:=;
end; procedure build(i,l,r:longint);
var m:longint;
begin
if l=r then read(tree[i])
else begin
m:=(l+r) div ;
build(i*,l,m);
build(*i+,m+,r);
tree[i]:=tree[*i]+tree[*i+];
end;
end; function find(i,l,r,l1,r1:longint):int64;
var m:longint;
t:int64;
begin
if (l>=l1) and (r<=r1) then exit(tree[i])
else begin
pushdown(l,r,i);
m:=(l+r) div ;
t:=;
if l1<=m then t:=t+find(*i,l,m,l1,r1);
if r1>m then t:=t+find(*i+,m+,r,l1,r1);
exit(t);
end;
end; procedure work(i,l,r,l1,r1,x:longint);
var m:longint;
begin
if (l1<=l) and (r<=r1) then
begin
lazy[i]:=lazy[i]+x;
tree[i]:=tree[i]+(r-l+)*x;
end
else begin
pushdown(l,r,i);
m:=(l+r) div ;
if (l1<=m) then work(i*,l,m,l1,r1,x);
if (r1>m) then work(i*+,m+,r,l1,r1,x);
tree[i]:=tree[i*]+tree[i*+];
end;
end; begin
readln(n,m);
build(,,n);
readln;
fillchar(lazy,sizeof(lazy),);
for i:= to m do
begin
read(c);
if c='Q' then
begin
read(a,b);
ans:=find(,,n,a,b);
writeln(ans);
end
else if c='C' then
begin
read(a,b,j);
work(,,n,a,b,j);
end;
readln;
end;
end.

poj3468

而poj2528还要复杂一点,简单的建立线段树会爆空间,这需要我们把出现的区间离散化,减小空间复杂度。

 var tree:array[..] of integer;
    x,y:array[..] of longint;
    a:array[..] of longint;         //表示离散化乎的标号对应的区间
    f:array[..] of boolean;
    ff:array[..] of boolean;
    i,j,k,n,t,s:longint;
procedure sort(l,r: longint);
  var i,j,x,y: longint;
  begin
    i:=l;
    j:=r;
    x:=a[(l+r) div ];
    repeat
      while a[i]<x do inc(i);
      while x<a[j] do dec(j);
      if not(i>j) then
      begin
        y:=a[i];
        a[i]:=a[j];
        a[j]:=y;
        inc(i);
        j:=j-;
      end;
    until i>j;
    if l<j then sort(l,j);
    if i<r then sort(i,r);
  end;
procedure putdown(i,p,q:longint);         //传递标记
  begin
    if p<>q then
    begin
      tree[i*]:=tree[i];
      tree[i*+]:=tree[i];
      tree[i]:=;
    end;
  end;
procedure build(i,p,q,l,r,x:longint);
  var m:longint;
  begin
    if (a[p]>=l) and (r>=a[q]) then tree[i]:=x
    else begin
      if tree[i]<> then putdown(i,p,q);
      m:=(p+q) div ;
      if l<=a[m] then
      begin
        build(i*,p,m,l,r,x);
      end;
      if r>a[m] then
      begin
        build(i*+,m+,q,l,r,x);
      end;
    end;
  end;
procedure dfs(i,p,q:longint);              //统计多少可见海报
  var m:longint;
  begin
    if (tree[i]>) and not ff[tree[i]] then
    begin
      s:=s+;
      ff[tree[i]]:=true;
    end
    else if (tree[i]=) and (p<>q) then
    begin
      m:=(p+q) div ;
      dfs(i*,p,m);
      dfs(i*+,m+,q);
    end;
  end;
begin
  readln(t);
  for i:= to t do
  begin
    k:=;
    fillchar(f,sizeof(f),false);
    readln(n);                        
    for j:= to n do 
    begin
      readln(x[j],y[j]);
      if not f[x[j]] then                        //离散化
      begin
        k:=k+;
        a[k]:=x[j];
        f[x[j]]:=true;
      end;
      if not f[y[j]] then
      begin
        k:=k+;
        a[k]:=y[j];
        f[y[j]]:=true;
      end;
    end;
    sort(,k);
    fillchar(tree,sizeof(tree),);
    for j:= to n do                     
      build(,,k,x[j],y[j],j);
    s:=;
    fillchar(ff,sizeof(ff),false);
    dfs(,,k);
    writeln(s);
  end;
end.

poj2528

poj3468,poj2528的更多相关文章

  1. 【poj3468】 A Simple Problem with Integers

    http://poj.org/problem?id=3468 (题目链接) 题意 给出一个序列,要求维护区间修改与区间求和操作. Solution 多年以前学习的树状数组区间修改又忘记了→_→. 其实 ...

  2. poj2528(线段树+离散化)

    题目链接:https://vjudge.net/problem/POJ-2528 题意:在区间[1,1e7]内染色,依次染n(<=1e4)中颜色,给出每种颜色染色的范围,可重叠,求最终有多少种颜 ...

  3. poj3468 线段树的懒惰标记

    题目链接:poj3468 题意:给定一段数组,有两种操作,一种是给某段区间加c,另一种是查询一段区间的和 思路:暴力的方法是每次都给这段区间的点加c,查询也遍历一遍区间,复杂度是n*n,肯定过不去,另 ...

  4. poj3468 A Simple Problem with Integers(线段树区间更新)

    https://vjudge.net/problem/POJ-3468 线段树区间更新(lazy数组)模板题 #include<iostream> #include<cstdio&g ...

  5. poj-2528线段树练习

    title: poj-2528线段树练习 date: 2018-10-13 13:45:09 tags: acm 刷题 categories: ACM-线段树 概述 这道题坑了我好久啊啊啊啊,,,, ...

  6. 线段树---poj3468 A Simple Problem with Integers:成段增减:区间求和

    poj3468 A Simple Problem with Integers 题意:O(-1) 思路:O(-1) 线段树功能:update:成段增减 query:区间求和 Sample Input 1 ...

  7. 线段树---poj2528 Mayor’s posters【成段替换|离散化】

    poj2528 Mayor's posters 题意:在墙上贴海报,海报可以互相覆盖,问最后可以看见几张海报 思路:这题数据范围很大,直接搞超时+超内存,需要离散化: 离散化简单的来说就是只取我们需要 ...

  8. POJ2528 Mayor's posters —— 线段树染色 + 离散化

    题目链接:https://vjudge.net/problem/POJ-2528 The citizens of Bytetown, AB, could not stand that the cand ...

  9. POJ3468 A Simple Problem with Integers —— 线段树 区间修改

    题目链接:https://vjudge.net/problem/POJ-3468 You have N integers, A1, A2, ... , AN. You need to deal wit ...

随机推荐

  1. c语言关于二进制的输出

    c语言中的二进制输出是没有占位符的,不像八进制:%o: 和十六进制:x%: c中二进制的输出 //右移31位,从最高为开始和1做&运算,得到每一位的二进制数值 void printbinry( ...

  2. Eclipse 3.7(代号Indigo) 中文字体太小解决办法(转)

    升级到3.7Eclipse最直观的反映就是,中文怎么那么小啊---- 相当不方便. 其实这是Eclipse的默认字体换了,以前的一直是Courier New,现在修改字体也找不到了,算了不找了. 这次 ...

  3. 2006: [NOI2010]超级钢琴 - BZOJ

    Description小Z是一个小有名气的钢琴家,最近C博士送给了小Z一架超级钢琴,小Z希望能够用这架钢琴创作出世界上最美妙的音乐. 这架超级钢琴可以弹奏出n个音符,编号为1至n.第i个音符的美妙度为 ...

  4. Leetcode#143 Reorder List

    原题地址 先把链表分割成前后两半,然后交叉融合 实践证明,凡是链表相关的题目,都应该当成工程类题目做,局部变量.功能函数什么的随便整,代码长了没关系,关键是清楚,不容易出错. 代码: ListNode ...

  5. Introduction to Deep Neural Networks

    Introduction to Deep Neural Networks Neural networks are a set of algorithms, modeled loosely after ...

  6. PAT-乙级-1051. 复数乘法 (15)

    1051. 复数乘法 (15) 时间限制 400 ms 内存限制 65536 kB 代码长度限制 8000 B 判题程序 Standard 作者 CHEN, Yue 复数可以写成(A + Bi)的常规 ...

  7. linux入门教程(九) 文本编辑工具vim

    前面多次提到过vim这个东西,它是linux中必不可少的一个工具.没有它很多工作都无法完成.早期的Unix都是使用的vi作为系统默认的编辑器的.你也许会有疑问,vi与vim有什么区别?可以这样简单理解 ...

  8. MAC 上升级python为最新版本

    第1步:下载Python3.4 下载地址如下: 下载Mac OS X 64-bit/32-bit installer https://www.python.org/downloads/release/ ...

  9. Oracle中关于数据库实例名与数据库服务名(转载)

    今天同事,出现了数据库连接失败的问题,一起百度了一下,结果总算解决了,以下是一些转载过来的普及知识. 1.查询数据库名:select name,dbid from v$database;或者命令行:s ...

  10. SpringMVC学习总结(六)——SpringMVC文件上传例子(2)

    基本的SpringMVC的搭建在我的上一篇文章里已经写过了,这篇文章主要说明一下使用SpringMVC进行表单上的文件上传以及多个文件同时上传的不同方法 一.配置文件: SpringMVC 用的是 的 ...