主体段树,要注意,因为有set和add操作,当慵懒的标志下推。递归优先set,后复发add,每次运行set行动add马克清0
WA了好几次是由于计算那一段的时候出问题了,可笑的是我对着模板找了一个多小时的错。
#include<cstdio>
#include<cmath>
#include<queue>
#include<stack>
#include<map>
#include<algorithm>
using namespace std;
#define lson pos << 1
#define rson pos << 1|1
typedef unsigned long long ULL;
typedef long long LL;
const int maxn = 1111111;
const int INF = 1 << 30;
struct Node{
int left,right;
int max_value,min_value,sum_value;
int col;
int col2;
}tr[maxn << 2];
struct Ans{
int min_value;
int max_value;
int sum_value;
Ans(int a = INF,int b = 0,int c = 0):min_value(a),max_value(b),sum_value(c){};
};
int n,m,k;
void pushdown(int pos){
if(tr[pos].col){
int len1 = tr[lson].right - tr[lson].left + 1;
int len2 = tr[rson].right - tr[rson].left + 1;
tr[lson].sum_value += (tr[pos].col * len1);
tr[lson].max_value += tr[pos].col;
tr[lson].min_value += tr[pos].col;
tr[rson].sum_value += (tr[pos].col * len2);
tr[rson].max_value += tr[pos].col;
tr[rson].min_value += tr[pos].col;
tr[lson].col += tr[pos].col;
tr[rson].col += tr[pos].col;
tr[pos].col = 0;
}
return ;
}
void pushdown2(int pos){
if(tr[pos].col2 >= 0){
int len1 = tr[lson].right - tr[lson].left + 1;
int len2 = tr[rson].right - tr[rson].left + 1;
tr[lson].sum_value = (tr[pos].col2 * len1);
tr[lson].max_value = tr[pos].col2;
tr[lson].min_value = tr[pos].col2;
tr[rson].sum_value = (tr[pos].col2 * len2);
tr[rson].max_value = tr[pos].col2;
tr[rson].min_value = tr[pos].col2;
tr[lson].col2 = tr[pos].col2; tr[lson].col = 0; //set递推须要清空求和
tr[rson].col2 = tr[pos].col2; tr[rson].col = 0;
tr[pos].col2 = -1;
}
return ;
}
void pushup(int pos){
tr[pos].sum_value = tr[lson].sum_value + tr[rson].sum_value;
tr[pos].max_value = max(tr[lson].max_value,tr[rson].max_value);
tr[pos].min_value = min(tr[lson].min_value,tr[rson].min_value);
return ;
}
void BuildTree(int L,int R,int pos){
tr[pos].left = L; tr[pos].right = R; tr[pos].col = 0;tr[pos].col2 = -1;
tr[pos].max_value = 0; tr[pos].min_value = 0; tr[pos].sum_value = 0;
if(L == R) return ;
int m = (L + R) >> 1;
BuildTree(L,m,lson);
BuildTree(m + 1,R,rson);
return ;
}
void TreeAdd(int l,int r,int add,int pos){
if(l <= tr[pos].left && tr[pos].right <= r){
int len = tr[pos].right - tr[pos].left + 1;
tr[pos].max_value += add; tr[pos].min_value += add;
tr[pos].sum_value += (add * len); tr[pos].col += add;
return ;
}
pushdown2(pos);
pushdown(pos);
int m = (tr[pos].left + tr[pos].right) >> 1;
if(l <= m)
TreeAdd(l,r,add,lson);
if(r > m)
TreeAdd(l,r,add,rson);
pushup(pos);
}
void TreeSet(int l,int r,int value,int pos){
if(l <= tr[pos].left && tr[pos].right <= r){
int len = tr[pos].right - tr[pos].left + 1;
tr[pos].max_value = value; tr[pos].min_value = value;
tr[pos].sum_value = (value * len); tr[pos].col2 = value;tr[pos].col = 0;
return ;
}
pushdown2(pos);
pushdown(pos);
int m = (tr[pos].left + tr[pos].right) >> 1;
if(l <= m)
TreeSet(l,r,value,lson);
if(r > m)
TreeSet(l,r,value,rson);
pushup(pos);
}
Ans Query(int l,int r,int pos){
if(l <= tr[pos].left && tr[pos].right <= r){
return Ans(tr[pos].min_value,tr[pos].max_value,tr[pos].sum_value);
}
pushdown2(pos);
pushdown(pos);
int m = (tr[pos].left + tr[pos].right) >> 1;
Ans a,b;
if(l <= m)
a = Query(l,r,lson);
if(r > m)
b = Query(l,r,rson);
pushup(pos);
return Ans(min(a.min_value,b.min_value),max(a.max_value,b.max_value),a.sum_value + b.sum_value);
}
int main(){
while(scanf("%d%d%d",&n,&m,&k) != EOF){
BuildTree(1,n * m,1);
while(k--){
int t;
scanf("%d",&t);
if(t == 1){
int x1,y1,x2,y2,v;//add
scanf("%d%d%d%d%d",&x1,&y1,&x2,&y2,&v);
for(int i = x1 ; i <= x2 ; i++){
int l = (i - 1) * m + y1;
int r = (i - 1) * m + y2;
TreeAdd(l,r,v,1);
}
}
else if(t == 2){
int x1,y1,x2,y2,v;//add
scanf("%d%d%d%d%d",&x1,&y1,&x2,&y2,&v);
for(int i = x1 ; i <= x2 ; i++){
int l = (i - 1) * m + y1;
int r = (i - 1) * m + y2;
TreeSet(l,r,v,1);
}
}
else if(t == 3){
int x1,y1,x2,y2;
Ans t;
int MAX = 0,MIN = INF,SUM = 0;
scanf("%d%d%d%d",&x1,&y1,&x2,&y2);
for(int i = x1 ; i <= x2 ; i++){
int l = (i - 1) * m + y1;
int r = (i - 1) * m + y2;
t = Query(l,r,1);
SUM += t.sum_value;
MAX = max(MAX,t.max_value);MIN = min(MIN,t.min_value);
}
printf("%d %d %d\n",SUM,MIN,MAX);
}
}
}
return 0;
}
版权声明:本文博客原创文章,博客,未经同意,不得转载。
【UVA】11992 - Fast Matrix Operations(段树模板)的更多相关文章
-
UVA 11992 - Fast Matrix Operations(段树)
UVA 11992 - Fast Matrix Operations 题目链接 题意:给定一个矩阵,3种操作,在一个矩阵中加入值a,设置值a.查询和 思路:因为最多20列,所以全然能够当作20个线段树 ...
-
uva 11992 Fast Matrix Operations 线段树模板
注意 setsetset 和 addvaddvaddv 标记的下传. 我们可以控制懒惰标记的优先级. 由于 setsetset 操作的优先级高于 addaddadd 操作,当下传 setsetset ...
-
UVA11992 - Fast Matrix Operations(段树部分的变化)
UVA11992 - Fast Matrix Operations(线段树区间改动) 题目链接 题目大意:给你个r*c的矩阵,初始化为0. 然后给你三种操作: 1 x1, y1, x2, y2, v ...
-
线段树(多维+双成段更新) UVA 11992 Fast Matrix Operations
题目传送门 题意:训练指南P207 分析:因为矩阵不超过20行,所以可以建20条线段的线段树,支持两个区间更新以及区间查询. #include <bits/stdc++.h> using ...
-
UVA 11992 Fast Matrix Operations(线段树:区间修改)
题目链接 2015-10-30 https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=s ...
-
UVA 11992 Fast Matrix Operations (二维线段树)
解法:因为至多20行,所以至多建20棵线段树,每行建一个.具体实现如下,有些复杂,慢慢看吧. #include <iostream> #include <cstdio> #in ...
-
UVa 11992 Fast Matrix Operations (线段树,区间修改)
题意:给出一个row*col的全0矩阵,有三种操作 1 x1 y1 x2 y2 v:将x1 <= row <= x2, y1 <= col <= y2里面的点全部增加v: 2 ...
-
uva 11992 - Fast Matrix Operations
简单的线段树的题: 有两种方法写这个题,目前用的熟是这种慢点的: 不过不知道怎么老是T: 感觉网上A过的人的时间度都好小,但他们都是用数组实现的 难道是指针比数组慢? 好吧,以后多用数组写写吧! 超时 ...
-
UVA 11992 Fast Matrix Operations (降维)
题意:对一个矩阵进行子矩阵操作. 元素最多有1e6个,树套树不好开(我不会),把二维坐标化成一维的,一个子矩阵操作分解成多条线段的操作. 一次操作的复杂度是RlogC,很容易找到极端的数据(OJ上实测 ...
随机推荐
-
Mysql 主从热备份
工作原理 首先锁定并备份主服务器数据库,从服务器导入备份的数据库,实现两个数据库的初态一样.然后把主服务器上执行过的sql语句都记录到二进制日志 Binarylog 中,从服务器会来读取这个log, ...
-
【uTenux实验】互斥体
互斥体,*中交互斥锁.其定义是这样的:互斥锁(英语:英语:Mutual exclusion,缩写 Mutex)是一种用于多线程编程中,防止两条线程同时对同一公共资源(比如全局变量)进行读写的机制 ...
-
AngularJS 1.5.0-beta.2 and 1.4.8 have been released
AngularJS 1.5.0-beta.2 With AngularJS 1.5.0-beta.2, we’ve improved the performance and flexibility o ...
-
[下载] VS 2013 Update 4 &; 社群版 (Visual Studio Community) &; VS 2015 Preview预览版
这是我的备份,原文请看http://www.dotblogs.com.tw/mis2000lab/archive/2014/11/13/vs2013_update4_community_vs2015_ ...
-
C#中 ? 和?? 的用法
C#中 ?? 和? 的意思 1.? 如果直接定义一个 值类型,给负值null:就会提示“无法将 Null转换成‘值类型(比如:int)’,因为他是一种不可为null的值 de类型” 例如 int in ...
-
谈谈怎么实现Oracle数据库分区表
谈谈怎么实现Oracle数据库分区表 数据库的读写分离 SQLSERVER性能监控级别步骤 Oracle索引问题诊断与优化(1)
-
项目总结之Oauth2.0免登陆及相关知识点总结
简介Oauth2.0授权步骤 授权码模式的基本步骤 原文链接地址 (A)用户访问客户端,后者将前者导向认证服务器. (B)用户选择是否给予客户端授权. (C)假设用户给予授权,认证服务器将用户导向客户 ...
-
Jmeter在非GUI环境下传递参数(命令行&;Jenkins配置)
https://www.cnblogs.com/kill0001000/p/8078686.html 通过cmd运行 jmeter -? 可以得到所有命令行选项(本文最后) 其中可以看到下面 -J 的 ...
-
springmvc aop 事务配置
对应的中文: 任意公共方法的执行: execution(public * *(..)) 任何一个以“set”开始的方法的执行: execution(* set*(..)) AccountService ...
-
HTML5 学习总结(三)——本地存储(localStorage、sessionStorage、WebSqlDataBase、IndexedDB)
HTML5问世以后,前端加入了一个重要的功能,便是本地存储,本地存储可分为4类: Local Storage:总的存储量有所限制,并不能提供真正的检索API,数据的生命期比窗口或浏览器的生命期长,数据 ...