思路
注意到最多20行,拆成20颗线段树即可
注意set标记清空左右儿子的add,不要清空自己的add,因为这个set操作之后可能还有add存在这个节点上
代码
#include <cstdio>
#include <algorithm>
#include <cstring>
#define int long long
using namespace std;
struct Node{
int minx,maxx,sum,add,set,lson,rson;
}Seg[5000000];
struct QNode{
int minx,maxx,sum;
};
int Nodecnt,r,c,m,root[30];
void pushup(int o){
Seg[o].sum=Seg[Seg[o].lson].sum+Seg[Seg[o].rson].sum;
Seg[o].maxx=max(Seg[Seg[o].lson].maxx,Seg[Seg[o].rson].maxx);
Seg[o].minx=min(Seg[Seg[o].lson].minx,Seg[Seg[o].rson].minx);
}
int New_Node(void){
int o=++Nodecnt;
Seg[o].add=Seg[o].lson=Seg[o].rson=Seg[o].maxx=Seg[o].minx=Seg[o].sum=0;
Seg[o].set=-1;
return o;
}
void pushdown(int o,int l,int r){
int mid=(l+r)>>1;
if(!Seg[o].lson)
Seg[o].lson=New_Node();
if(!Seg[o].rson)
Seg[o].rson=New_Node();
if(Seg[o].set!=-1){
Seg[Seg[o].lson].set=Seg[o].set;
Seg[Seg[o].rson].set=Seg[o].set;
Seg[Seg[o].lson].sum=Seg[o].set*(mid-l+1);
Seg[Seg[o].rson].sum=Seg[o].set*(r-mid);
Seg[Seg[o].lson].maxx=Seg[o].set;
Seg[Seg[o].rson].maxx=Seg[o].set;
Seg[Seg[o].lson].minx=Seg[o].set;
Seg[Seg[o].rson].minx=Seg[o].set;
Seg[Seg[o].lson].add=0;
Seg[Seg[o].rson].add=0;
Seg[o].set=-1;
}
if(Seg[o].add){
Seg[Seg[o].lson].add+=Seg[o].add;
Seg[Seg[o].rson].add+=Seg[o].add;
Seg[Seg[o].lson].sum+=Seg[o].add*(mid-l+1);
Seg[Seg[o].rson].sum+=Seg[o].add*(r-mid);
Seg[Seg[o].lson].maxx+=Seg[o].add;
Seg[Seg[o].rson].maxx+=Seg[o].add;
Seg[Seg[o].lson].minx+=Seg[o].add;
Seg[Seg[o].rson].minx+=Seg[o].add;
Seg[o].add=0;
}
}
void add(int L,int R,int l,int r,int &o,int c){
if(!o)
o=New_Node();
if(L<=l&&r<=R){
Seg[o].add+=c;
Seg[o].maxx+=c;
Seg[o].minx+=c;
Seg[o].sum+=c*(r-l+1);
return;
}
pushdown(o,l,r);
int mid=(l+r)>>1;
if(L<=mid)
add(L,R,l,mid,Seg[o].lson,c);
if(R>mid)
add(L,R,mid+1,r,Seg[o].rson,c);
pushup(o);
}
void set(int L,int R,int l,int r,int &o,int c){
if(!o)
o=New_Node();
if(L<=l&&r<=R){
Seg[o].add=0;
Seg[o].set=c;
Seg[o].maxx=c;
Seg[o].minx=c;
Seg[o].sum=c*(r-l+1);
return;
}
pushdown(o,l,r);
int mid=(l+r)>>1;
if(L<=mid)
set(L,R,l,mid,Seg[o].lson,c);
if(R>mid)
set(L,R,mid+1,r,Seg[o].rson,c);
pushup(o);
}
QNode query(int L,int R,int l,int r,int &o){
if(!o)
o=New_Node();
QNode tmp;
if(L<=l&&r<=R){
tmp.maxx=Seg[o].maxx;
tmp.minx=Seg[o].minx;
tmp.sum=Seg[o].sum;
return tmp;
}
pushdown(o,l,r);
int mid=(l+r)>>1;
if(R<=mid)
return query(L,R,l,mid,Seg[o].lson);
else if(L>mid)
return query(L,R,mid+1,r,Seg[o].rson);
else{
QNode lx,rx;
lx=query(L,R,l,mid,Seg[o].lson);
rx=query(L,R,mid+1,r,Seg[o].rson);
tmp.minx=min(lx.minx,rx.minx);
tmp.maxx=max(lx.maxx,rx.maxx);
tmp.sum=lx.sum+rx.sum;
return tmp;
}
}
void init(void){
Nodecnt=0;
memset(root,0,sizeof(root));
Seg[0].maxx=-0x3f3f3f3f;
Seg[0].minx=0x3f3f3f3f;
Seg[0].sum=0;
}
signed main(){
// freopen("test.in","r",stdin);
// freopen("test.out","w",stdout);
while(scanf("%lld %lld %lld",&r,&c,&m)==3){
init();
int x1,y1,x2,y2,v,opt;
for(int i=1;i<=m;i++){
scanf("%lld %lld %lld %lld %lld",&opt,&x1,&y1,&x2,&y2);
if(opt==1){
scanf("%lld",&v);
for(int j=x1;j<=x2;j++)
add(y1,y2,1,c,root[j],v);
}
else if(opt==2){
scanf("%lld",&v);
for(int j=x1;j<=x2;j++)
set(y1,y2,1,c,root[j],v);
}
else{
QNode ans={0x3f3f3f3f,-0x3f3f3f3f,0},tmp;
for(int j=x1;j<=x2;j++){
tmp=query(y1,y2,1,c,root[j]);
ans.maxx=max(ans.maxx,tmp.maxx);
ans.minx=min(ans.minx,tmp.minx);
ans.sum=ans.sum+tmp.sum;
}
printf("%lld %lld %lld
",ans.sum,ans.minx,ans.maxx);
}
}
// printf("0:%lld %lld %lld
",Seg[0].maxx,Seg[0].minx,Seg[0].sum);
}
return 0;
}