题目

“第一分钟,X说,要有矩阵,于是便有了一个里面写满了0的n×m矩阵。

第二分钟,L说,要能修改,于是便有了将左上角为(a,b),右下角为(c,d)的一个矩形区域内的全部数字加上一个值的操作。

第三分钟,k说,要能查询,于是便有了求给定矩形区域内的全部数字和的操作。

第四分钟,彩虹喵说,要基于二叉树的数据结构,于是便有了数据范围。

第五分钟,和雪说,要有耐心,于是便有了时间限制。

第六分钟,吃钢琴男说,要省点事,于是便有了保证运算过程中及最终结果均不超过32位有符号整数类型的表示范围的限制。

第七分钟,这道题终于造完了,然而,造题的神牛们再也不想写这道题的程序了。”

   ——《上帝造裸题的七分钟》

所以这个神圣的任务就交给你了。

输入格式

输入数据的第一行为X n m,代表矩阵大小为n×m。

从输入数据的第二行开始到文件尾的每一行会出现以下两种操作:

  L a b c d delta —— 代表将(a,b),(c,d)为顶点的矩形区域内的所有数字加上delta。

  k a b c d   —— 代表求(a,b),(c,d)为顶点的矩形区域内所有数字的和。

请注意,k为小写。

输出格式

针对每个k操作,在单独的一行输出答案。

输入样例

X 4 4

L 1 1 3 3 2

L 2 2 4 4 1

k 2 2 3 3

输出样例

12

范围

对于100%的数据,1 ≤ n ≤ 2048, 1 ≤ m ≤ 2048, 1 ≤ abs(delta) ≤ 500,操作不超过200000个,保证运算过程中及最终结果均不超过32位带符号整数类型的表示范围。

题解

一开始想写个二维线段树,写到一半发现第一维修改时的合并操作很难实现。。于是改用树状数组

二维树状数组

首先我们要先了解二维树状数组的作用

二维树状数组,故名思议,就是二维的树状数组,可以维护二维的前缀和,具体实现类似一维:

struct BIT{
int A[maxn][maxn];
void add(int x,int y,int v){
for (int i = x; i <= n; i += lbt(i))
for (int j = y; j <= m; j += lbt(j))
A[i][j] += v;
}
int sum(int x,int y){
int ans = 0;
for (int i = x; i > 0; i -= lbt(i))
for (int j = y; j > 0; j -= lbt(j))
ans += A[i][j];
return ans;
}
};

它维护的是前缀和,单点修改。



假如我们有了这样一个数据结构,想求(x1,y1)~(x2,y2)的和

那么sum = sum(x2,y2) - sum(x2,y1 - 1) - sum(x1 - 1,y2) + sum(x1 - 1,y1 - 1)

就像容斥一样,很直观的说

我们现在就有了一个单点修改,二维区间求和的工具

单次修改或查询复杂度O(log2n)

区间修改的树状数组

树状数组如何实现区间修改?

我们从最简单的一维树状数组入手:

众所周知,一维树状数组是单点修改维护前缀和而实现区间求和

假若我们想使用单点修改 + 前缀和 实现区间修改,你想到了什么?

差分数组!

差分数组每次在区间的两端点进行修改,求一次前缀和即可还原这个数组

假若我们用树状数组维护一个差分数组,那岂不是可以做到区间修改?

没错,是的,可以做到区间修改

但是你有没有想过你把区间和丢哪了?

我们维护一个差分数组,对i位置取前缀和,得到的是i位置的值,并不能得到前缀和

假若在差分数组中,我们要求前缀和:

sum(i,j)=D[1∼i]+D[1∼i+1]+D[1∼i+2]+...+D[1∼j]

=(j−i+1)∗∑ik=1D[k]+(j−i)∗D[i+1]+(j−i−1)∗D[i+2]+....+D[j]

=(j+1)∗∑jk=iD[k]−(j∗D[j]+(j−1)∗D[j−1]+(j−2)∗D[j−2]+...+i∗D[i])

=(j+1)∗SUM(D[x],i,j)−SUM(D[x]∗x,i,j)

所以,我们只用维护D[i]的前缀和和D[i]*i的前缀和就可以算出区间和了

二维树状数组

扩展到二维也是类似的,我们令A[i][j]表示(i,j)到(n,m)的增量



可以自己yy一下,写博客真累

放代码【很短的】:

#include<cstdio>
#define lbt(u) (u & -u)
const int maxn = 2050,maxm = 100005,INF = 1000000000;
inline int RD(){
int out = 0,flag = 1; char c = getchar();
while (c < 48 || c > 57) {if (c == '-') flag = -1; c = getchar();}
while (c >= 48 && c <= 57) {out = (out << 1) + (out << 3) + c - '0'; c = getchar();}
return out * flag;
}
int n,m;
struct BIT{
int A[maxn][maxn];
void add(int x,int y,int v){
for (int i = x; i <= n; i += lbt(i))
for (int j = y; j <= m; j += lbt(j))
A[i][j] += v;
}
int sum(int x,int y){
int ans = 0;
for (int i = x; i > 0; i -= lbt(i))
for (int j = y; j > 0; j -= lbt(j))
ans += A[i][j];
return ans;
}
}A,B,C,D;
void Add(int x,int y,int v){
A.add(x,y,v);
B.add(x,y,y * v);
C.add(x,y,x * v);
D.add(x,y,x * y * v);
}
void update(int x1,int y1,int x2,int y2,int v){
Add(x1,y1,v); Add(x2 + 1,y1,-v); Add(x1,y2 + 1,-v); Add(x2 + 1,y2 + 1,v);
}
int Sum(int x,int y){
return (x + 1) * (y + 1) * A.sum(x,y) - (x + 1) * B.sum(x,y) - (y + 1) * C.sum(x,y) + D.sum(x,y);
}
int Query(int x1,int y1,int x2,int y2){
return Sum(x2,y2) - Sum(x1 - 1,y2) - Sum(x2,y1 - 1) + Sum(x1 - 1,y1 - 1);
}
int main(){
n = RD(); m = RD(); int x1,y1,x2,y2,d;
while (true){
char c = getchar();
while (c != EOF && c != 'L' && c != 'k') c = getchar();
if (c == EOF) break;
else if (c == 'L'){
x1 = RD(); y1 = RD(); x2 = RD(); y2 = RD(); d = RD();
update(x1,y1,x2,y2,d);
}else {
x1 = RD(); y1 = RD(); x2 = RD(); y2 = RD();
printf("%d\n",Query(x1,y1,x2,y2));
}
}
return 0;
}

BZOJ3132 上帝造题的七分钟 【二维树状数组】的更多相关文章

  1. [bzoj3132]上帝造题的七分钟——二维树状数组

    题目大意 你需要实现一种数据结构,支援以下操作. 给一个矩阵的子矩阵的所有元素同时加一个数. 计算子矩阵和. 题解 一看这个题,我就首先想到用线段树套线段树做. 使用二维线段树的错误解法 其实是第一次 ...

  2. 【bzoj3132】上帝造题的七分钟 二维树状数组区间修改区间查询

    题目描述 “第一分钟,X说,要有矩阵,于是便有了一个里面写满了0的n×m矩阵. 第二分钟,L说,要能修改,于是便有了将左上角为(a,b),右下角为(c,d)的一个矩形区域内的全部数字加上一个值的操作. ...

  3. 【BZOJ3132】【TYVJ1716】上帝造题的七分钟 二维树状数组

    题目大意 维护一个\(n\times m\)的矩阵,有两种操作: \(1~x_1~y_1~x_2~y_2~v\):把\((a,b),(c,d)\)为顶点的矩形区域内的所有数字加上\(v\). \(2~ ...

  4. tyvj P1716 - 上帝造题的七分钟 二维树状数组区间查询及修改 二维线段树

    P1716 - 上帝造题的七分钟 From Riatre    Normal (OI)总时限:50s    内存限制:128MB    代码长度限制:64KB 背景 Background 裸体就意味着 ...

  5. P4514 上帝造题的七分钟——二维树状数组

    P4514 上帝造题的七分钟 求的是矩阵里所有数的和: 维护四个树状数组: #include<cstdio> #include<cstring> #include<alg ...

  6. BZOJ 3132: 上帝造题的七分钟( 二维BIT )

    二维树状数组... 自己YY一下再推一下应该可以搞出来... --------------------------------------------------------------------- ...

  7. POJ 2029 Get Many Persimmon Trees (模板题)【二维树状数组】

    <题目链接> 题目大意: 给你一个H*W的矩阵,再告诉你有n个坐标有点,问你一个w*h的小矩阵最多能够包括多少个点. 解题分析:二维树状数组模板题. #include <cstdio ...

  8. P4514 上帝造题的七分钟(二维树状数组)

    P4514 上帝造题的七分钟 二维树状数组 差分维护区间加法,区间求和 #include<cstdio> int read(){ ,f=; ') f=f&&(c!='-') ...

  9. POJ 2155 Matrix (二维树状数组)

    Matrix Time Limit: 3000MS   Memory Limit: 65536K Total Submissions: 17224   Accepted: 6460 Descripti ...

  10. POJ_2155 Matrix 【二维树状数组】

    一.题面 POJ2155 二.分析 楼教主出的题,是二维树状数组非常好的题,还结合了开关问题(开关变化的次数如果为偶数,状态不变,奇数状态相反). 题意就是给了一个二维的坐标平面,每个点初始值都是0, ...

随机推荐

  1. webpack 4.14配置详解

    1.安装nodejs 官网下载nodejs,安装时可能会爆 2503错误,解决办法是:使用管理员命令执行安装文件.cmd ->命令提示符(管理员)-> 输入: msiexec /packa ...

  2. 更新Composer依赖报错处理Fatal error: Declaration of Fxp\Composer\AssetPlugin\Repository\AbstractAssetsRe

    更新Composer依赖报错处理 Fatal error: Declaration of Fxp\Composer\AssetPlugin\Repository\AbstractAssetsRe po ...

  3. Could not obtain transaction-synchronized Session for current thread 错误的解决方法!

    BsTable bsTable = new BsTable(); // String time = request.getParameter("date"); String tim ...

  4. Python学习手册之函数和模块

    在上一篇文章中,我们介绍了 Python 的控制结构,现在我们介绍 Python 函数和模块. 查看上一篇文章请点击:https://www.cnblogs.com/dustman/p/9976234 ...

  5. semcms 网站漏洞挖掘过程与安全修复防范

    emcms是国内第一个开源外贸的网站管理系统,目前大多数的外贸网站都是用的semcms系统,该系统兼容许多浏览器,像IE,google,360极速浏览器都能非常好的兼容,官方semcms有php版本, ...

  6. 幸运三角形 南阳acm491(dfs)

    幸运三角形 时间限制:1000 ms  |  内存限制:65535 KB 难度:3   描述 话说有这么一个图形,只有两种符号组成(‘+’或者‘-’),图形的最上层有n个符号,往下个数依次减一,形成倒 ...

  7. 深入了解jQuery Mobile-1

    介绍 jQuery Mobile是一种触控优化的HTML5 UI框架,旨在制作可在所有智能手机,平板电脑和台式机设备上访问的响应式网站和应用程序 移动页面结构 jQuery Mobile站点必须以HT ...

  8. windows 安装 .net core 环境

    windows 安装 环境说明 window10系统 .net core 1.0.1 visual studio code 安装 .net core Windows系统下安装软件基本上属于傻瓜式安装, ...

  9. Create Fiori List App Report with ABAP CDS view – PART 2

    In the Part 1 blog, we have discussed below topics CDS annotations for Fiori List Report. How to cre ...

  10. git 本地分支与远程分支 新建/删除/合并

    github上已经有master分支 和dev分支 在本地 git checkout -b dev 新建并切换到本地dev分支 git pull origin dev 本地分支与远程分支相关联 在本地 ...