这个题目要求和 还有 设置区间值 区间增值,明显要用线段树来

由于行数不超过20 而列数多达 10^5,所以对每一行建一棵线段树。

然后主要是在懒惰标记方面是难点 针对两种操作 分别设置 set 和 add 方法,但是优先级方面要好好考虑

可能出现的结果无非是 单独的 set 或者 add  以及 先set再add  或者 先add再set,单独的那两种就按常规写法,然后先set再add,则明显 add标记不能清除set,所以在pushdown的时候 先set 再 add,先set 再add ,则直接清除当前的add标记,在pushdown的时候同样清除子节点的add标记

其实只要搞清优先级 也挺简单的。。。懒惰标记用的还不熟练,要继续加强

#include <iostream>
#include <cstdio>
#include <cstring>
#define lson rt<<1,l,mid
#define rson rt<<1|1,mid+1,r
using namespace std;
const int N = +;
int d[][N*],dmax[][N*],dmin[][N*],setv[][N*],addv[][N*];
int r,c,m;
struct node
{
int tot,maxn,mini;
};
void up(int num,int rt)
{
d[num][rt]=d[num][rt<<]+d[num][rt<<|];
dmax[num][rt]=max(dmax[num][rt<<],dmax[num][rt<<|]);
dmin[num][rt]=min(dmin[num][rt<<],dmin[num][rt<<|]);
}
void build(int num,int rt,int l,int r)
{
setv[num][rt]=-;
addv[num][rt]=;
d[num][rt]=dmax[num][rt]=dmin[num][rt]=;
if (l>=r){
return;
}
int mid=(l+r)>>;
build(num,lson);
build(num,rson);
}
void pushdown(int num,int rt,int l,int r)
{
if (l>=r) return;
int mid=(l+r)>>;
if (setv[num][rt]>=){
d[num][rt<<]=(mid-l+)*setv[num][rt];
dmax[num][rt<<]=setv[num][rt];
dmin[num][rt<<]=setv[num][rt]; d[num][rt<<|]=(r-mid)*setv[num][rt];
dmax[num][rt<<|]=setv[num][rt];
dmin[num][rt<<|]=setv[num][rt]; setv[num][rt<<]=setv[num][rt<<|]=setv[num][rt];
setv[num][rt]=-;
addv[num][rt<<]=addv[num][rt<<|]=;
}
if (addv[num][rt]){
d[num][rt<<]+=(mid-l+)*addv[num][rt];
dmax[num][rt<<]+=addv[num][rt];
dmin[num][rt<<]+=addv[num][rt]; d[num][rt<<|]+=(r-mid) *addv[num][rt];
dmax[num][rt<<|]+=addv[num][rt];
dmin[num][rt<<|]+=addv[num][rt]; addv[num][rt<<]+=addv[num][rt];
addv[num][rt<<|]+=addv[num][rt];
addv[num][rt]=;
}
}
void pushdown2(int num,int rt,int l,int r)
{
if (l>=r) return;
int mid=(l+r)>>;
if (setv[num][rt]>=){
d[num][rt<<]=(mid-l+)*setv[num][rt];
dmax[num][rt<<]=setv[num][rt];
dmin[num][rt<<]=setv[num][rt]; d[num][rt<<|]=(r-mid)*setv[num][rt];
dmax[num][rt<<|]=setv[num][rt];
dmin[num][rt<<|]=setv[num][rt]; setv[num][rt<<]=setv[num][rt<<|]=setv[num][rt];
setv[num][rt]=-;
addv[num][rt<<]=addv[num][rt<<|]=;
}
if (addv[num][rt]){
d[num][rt<<]+=(mid-l+)*addv[num][rt];
dmax[num][rt<<]+=addv[num][rt];
dmin[num][rt<<]+=addv[num][rt]; d[num][rt<<|]+=(r-mid) *addv[num][rt];
dmax[num][rt<<|]+=addv[num][rt];
dmin[num][rt<<|]+=addv[num][rt]; addv[num][rt<<]+=addv[num][rt];
addv[num][rt<<|]+=addv[num][rt];
addv[num][rt]=;
}
} void add(int num,int v,int L,int R,int rt,int l,int r)
{ if (L<=l && r<=R){
d[num][rt]+=v*(r-l+);
dmax[num][rt]+=v;
dmin[num][rt]+=v;
addv[num][rt]+=v;
return;
}
pushdown(num,rt,l,r);
int mid=(l+r)>>;
if (L<=mid) add(num,v,L,R,lson);
if (R>mid) add(num,v,L,R,rson);
up(num,rt);
}
void sets(int num,int v,int L,int R,int rt,int l,int r)
{ if (L<=l && r<=R){
d[num][rt]=v*(r-l+);
dmax[num][rt]=v;
dmin[num][rt]=v;
setv[num][rt]=v;
addv[num][rt]=; //set标记直接清除当前的add标记 这里要注意别漏掉
return;
}
pushdown(num,rt,l,r);
int mid=(l+r)>>;
if (L<=mid) sets(num,v,L,R,lson);
if (R>mid) sets(num,v,L,R,rson);
up(num,rt);
}
node query(int num,int L,int R,int rt,int l,int r)
{
int mid=(l+r)>>;
pushdown2(num,rt,l,r);
if (L<=l && r<=R){
node tmp=(node){d[num][rt],dmax[num][rt],dmin[num][rt]};
return tmp;
}
node t1=(node){-,,};
node t2=(node){-,,};
if (L<=mid) t1=query(num,L,R,lson);
if (R>mid) t2=query(num,L,R,rson);
if (t1.tot<) return t2;
if (t2.tot<) return t1;
t1.tot+=t2.tot;
t1.maxn=max(t2.maxn,t1.maxn);
t1.mini=min(t1.mini,t2.mini);
return t1; }
int main()
{
while (scanf("%d%d%d",&r,&c,&m)!=EOF)
{
for (int i=;i<r;i++){
build(i,,,c);
}
while (m--){
int deno,x1,y1,x2,y2,v;
scanf("%d",&deno);
if (deno<=){
scanf("%d%d%d%d%d",&x1,&y1,&x2,&y2,&v);
x1--;x2--;
for (int i=x1;i<=x2;i++){
if (deno==) add(i,v,y1,y2,,,c);
else sets(i,v,y1,y2,,,c);
}
}
else{
scanf("%d%d%d%d",&x1,&y1,&x2,&y2);
x1--;x2--;
node ans;
for (int i=x1;i<=x2;i++){
if (i==x1) ans=query(i,y1,y2,,,c);
else {
node tmp=query(i,y1,y2,,,c);
ans.tot+=tmp.tot;
ans.maxn=max(tmp.maxn,ans.maxn);
ans.mini=min(tmp.mini,ans.mini);
}
}
printf("%d %d %d\n",ans.tot,ans.mini,ans.maxn);
}
}
}
return ;
}

UVA 11992 懒惰标记应用的更多相关文章

  1. 【HDU 4614】Vases and Flowers(线段树区间更新懒惰标记)

    题目0到n-1的花瓶,操作1在下标a开始插b朵花,输出始末下标.操作2清空[a,b]的花瓶,求清除的花的数量.线段树懒惰标记来更新区间.操作1,先查询0到a-1有num个空瓶子,然后用线段树的性质,或 ...

  2. bzoj1251 序列终结者(Splay Tree+懒惰标记)

    Description 网上有许多题,就是给定一个序列,要你支持几种操作:A.B.C.D.一看另一道题,又是一个序列 要支持几种操作:D.C.B.A.尤其是我们这里的某人,出模拟试题,居然还出了一道这 ...

  3. 【POJ】3468 A Simple Problem with Integers ——线段树 成段更新 懒惰标记

    A Simple Problem with Integers Time Limit:5000MS   Memory Limit:131072K Case Time Limit:2000MS Descr ...

  4. 【HDU】4092 Nice boat(多校第四场1006) ——线段树 懒惰标记

    Nice boat Time Limit: 30000/15000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others) To ...

  5. UVA 11992 - Fast Matrix Operations(段树)

    UVA 11992 - Fast Matrix Operations 题目链接 题意:给定一个矩阵,3种操作,在一个矩阵中加入值a,设置值a.查询和 思路:因为最多20列,所以全然能够当作20个线段树 ...

  6. poj3468 线段树的懒惰标记

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

  7. FZU-1608 Huge Mission 线段树(更新懒惰标记)

    题目链接: https://cn.vjudge.net/problem/FZU-1608 题目大意: 长度n,m次操作:每次操作都有三个数:a,b,c:意味着(a,b]区间单位长度的价值为c,若某段长 ...

  8. hdu1698 Just a Hook (线段树区间更新 懒惰标记)

    Just a Hook Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Tota ...

  9. HDU 4107 Gangster(线段树 特殊懒惰标记)

    两种做法. 第一种:标记区间最大值和最小值,若区间最小值>=P,则本区间+2c,若区间最大值<P,则本区间+c.非常简单的区间更新. 最后发一点牢骚:最后query查一遍就行,我这个2B竟 ...

随机推荐

  1. php 打印格式化显示利器 <pre>

    当我们PHP调试的时候,用var_dump 或 print_r打印json数据或array数组时,html页面没有换行显示,看到的内容一大堆,不好定位. 输出前添加 <pre>,便可以自动 ...

  2. C++ Primer Plus 6 笔记(2)

    第4章 1.求数组元素个数的一种方法:num=sizeof 数组名/sizeof (元素类型) 2.确定字符串所需的最短数组时,别忘了将结尾的'\0'计算在内.表面可以无,内存必须有. 3.'S'表示 ...

  3. 2.Jsoup

    public static void main(String[] args) { //爬取最大资源网上的数据 //用CSS选择器 try { Document doc = Jsoup.parse(ne ...

  4. Lua 完美打印数据 (例子)

    例子1 : ableprint = function(data,cstring,deepIndex) --第二个参数可以为空,第三个参数不要手动添加,它是用来进行打印深度控制的. if data == ...

  5. Luogu P3263 [JLOI2015]有意义的字符串

    Link 设\(e=\frac{b+\sqrt d}2,i=\frac{b-\sqrt d}2\). 显然\(f_n=e^n+i^n\)是一个整数,且\(f_n=(e+i)f_{n-1}+eif_{n ...

  6. Timer(阿里CTF)

    下载文件之后发现是.apk类型文件,于是百度搜索了一下推荐再电脑上安装安卓模拟器,都尝试了一下逍遥安卓是真的不好用,小白觉得还是BlueStacks好用一些,下载之后去安装打开就看到了, 什么都没有发 ...

  7. Java中遍历 Session 和 Request

    转: session的遍历: java.util.Enumeration e = request.getSession().getAttributeNames(); while( e.hasMoreE ...

  8. spoj694--Distinct Substrings

    个人第一道后缀数组题目.对于每一个后缀suffix(i),都有len-sa[i]个前缀(也即有len-sa[i]个不同的字符串),其中与排名前一位的后缀有height[i]个共同的前缀,最后所得到的新 ...

  9. 数据结构第二版之(课后题)BF算法病毒感染检测

    //vs2013下编译通过.换别的编译器自行补充头文件和修改源代码#include<iostream> #include<fstream> #include <strin ...

  10. TS文件极简合并

    TS文件是可以直接通过二进制拷贝连接的方式进行合并的,一般采用如下的命令行参数:copy /b 1.ts+2.ts+3.ts new.ts 这个例子就是将1.ts.2.ts.3.ts三个文件按顺序连接 ...