[bzoj3132]上帝造题的七分钟——二维树状数组
题目大意
你需要实现一种数据结构,支援以下操作。
- 给一个矩阵的子矩阵的所有元素同时加一个数。
- 计算子矩阵和。
题解
一看这个题,我就首先想到用线段树套线段树做。
使用二维线段树的错误解法
其实是第一次写二维线段树orz。为了方便,我们不再使用k<<1作为左儿子,k<<1|1作为右儿子,而是使用一个lc,rc数组来存储左孩子和右孩子。实现起来不是很麻烦但是有许多细节。我们可以在操作函数中写一个参数表示是操作的行还是操作的列。另外注意打标记。对于节点k,标记一旦下传,他的孩子的各种标记即被修改,而它自身的标记应该在更前面的一个时刻修改。同时注意update,因为这个WA了好多次。尽管这样,还是有许多错。小数据基本可以过,但大数据就不行了Orz,求各位大佬帮助修改。
使用二维树状数组的正解
首先我们先要把树状数组(单点修改)扩展的二维。很显然,我们可以直接在两个维度重现一维的操作,非常简单。
那么,对于区间修改,我们也要扩展到二维,我们回忆一维时候的做法,我们首先分类讨论,比较了修改一个区间对于一个前缀的影响,具体可以见《挑战程序设计竞赛》或者我的模板汇总贴。
我们继续这样考虑。记\(g(i,j)\)为修改后的二维前缀和,\(s(i,j)\)为修改前的二维前缀和。我们分类讨论,容易得出以下结论。
\]
\]
\]
\]
\]
上面的五个式子分别对应ij均小,ij均ok,i好j大,i大j好,i大j大五种情况。//上面的狮子\(x_0y_0\)均-1的原因是本蒟蒻开始推的时候忘记把区间前界-1了Orz
推演到这里就非常方便了。我们无论是使用差分的思想,还是使用乱搞的思想都非常好搞了。
具体地,我们开四个树状数组,分别表示与i,j均无关,与ij一个有关,与ij均有关四种情况,分清楚每一种情况的作用域,准确地运用差分即可。
其实,程序实现的时候有很多细节,一个是因为推导时候的错误,一个是因为树状数组从1开始存储,所以要把每个坐标都加一。
这个题写了一晚上,但AC了还是非常有成就感QAQ
代码(使用二维线段树的错误解答)
#include <algorithm>
#include <cstdio>
const int maxm = 3000;
const int maxn = 2048 * 2048 * 4;
int sum[maxn], ll[maxn], rr[maxn], add[maxn], lc[maxn], rc[maxn], sz = 0,
gg[maxm];
char wtf;
int n, m;
void pushdown(int k) {
if (add[k]) {
add[lc[k]] += add[k];
add[rc[k]] += add[k];
int mid = (ll[k] + rr[k]) >> 1;
sum[lc[k]] += (mid - ll[k] + 1) * (add[k]);
sum[rc[k]] += (rr[k] - mid) * (add[k]);
add[k] = 0;
}
return;
}
void update(int k) { sum[k] = sum[lc[k]] + sum[rc[k]]; }
int build(int k, int l, int r, int a) {
ll[k] = l, rr[k] = r;
if (l == r) {
if (a == 0) {
gg[k] = ++sz;
build(gg[k], 1, m, 1);
} else if (a == 1) {
sum[k] = 0;
}
return k;
}
int mid = (l + r) >> 1, left = build(++sz, l, mid, a),
right = build(++sz, mid + 1, r, a);
lc[k] = left, rc[k] = right;
sum[k] = sum[lc[k]] + sum[rc[k]];
return k;
}
void change(int k, int y0, int y1, int val) {
int l = ll[k], r = rr[k], mid = (l + r) >> 1;
if (y0 <= l && r <= y1) {
sum[k] += (r - l + 1) * (val);
add[k] = val;
return;
}
pushdown(k);
if (y0 <= mid)
change(lc[k], y0, y1, val);
if (y1 > mid)
change(rc[k], y0, y1, val);
update(k);
}
void plus(int k, int x0, int x1, int y0, int y1, int val) {
int l = ll[k], r = rr[k], mid = (l + r) >> 1;
if (x0 <= l && r <= x1 && l == r) {
change(gg[k], y0, y1, val);
return;
}
if (x0 <= mid)
plus(lc[k], x0, x1, y0, y1, val);
if (x1 > mid)
plus(rc[k], x0, x1, y0, y1, val);
update(k);
return;
}
int query(int k, int x0, int x1, int y0, int y1, int a) {
int l = ll[k], r = rr[k], mid = (l + r) >> 1;
if (a == 0) {
if (x0 <= l && r <= x1 && l == r) {
return query(gg[k], x0, x1, y0, y1, 1);
}
int ans = 0;
if (x0 <= mid)
ans += query(lc[k], x0, x1, y0, y1, 0);
if (x1 > mid)
ans += query(rc[k], x0, x1, y0, y1, 0);
return ans;
}
if (a == 1) {
if (y0 <= l && r <= y1) {
return sum[k];
}
pushdown(k);
int ans = 0;
if (y0 <= mid)
ans += query(lc[k], x0, x1, y0, y1, 1);
if (y1 > mid)
ans += query(rc[k], x0, x1, y0, y1, 1);
update(k);
return ans;
}
}
int main() {
#ifdef D
freopen("input", "r", stdin);
#endif
scanf("%c %d %d", &wtf, &n, &m);
int t = build(++sz, 1, n, 0);
char ch;
while (scanf("%c", &ch) == 1) {
if (ch == 'L') {
int x0, y0, x1, y1, val;
scanf("%d %d %d %d %d", &x0, &y0, &x1, &y1, &val);
if (x0 > x1) {
std::swap(x0, x1);
std::swap(y0, y1);
}
plus(t, x0, x1, y0, y1, val);
}
if (ch == 'k') {
int x0, y0, x1, y1;
scanf("%d %d %d %d", &x0, &y0, &x1, &y1);
if (x0 > x1 && y0 < y1) {
std::swap(y0, y1);
}
if (x0 > x1)
std::swap(x0, x1);
int ans = query(t, x0, x1, y0, y1, 0);
printf("%d\n", ans);
}
}
return 0;
}
代码(使用树状数组的正解)
#include <algorithm>
#include <cstdio>
#include <cstring>
#include <iostream>
#ifdef D
const int maxn = 30;
#else
const int maxn = 2052;
#endif
int n, m;
int c
[4][maxn]
[maxn]; // a:与i无关与j无关,b:与i有关与j无关,c:与i无关与j有关,d:与i有关与j有关
char opt;
void change(int id, int x, int y, int val) {
for (int i = x; i <= n; i += i & -i) {
for (int j = y; j <= m; j += j & -j) {
c[id][i][j] += val;
}
}
}
int qu(int id, int x, int y) {
int ans = 0;
for (int i = x; i > 0; i -= i & -i) {
for (int j = y; j > 0; j -= j & -j) {
ans += c[id][i][j];
}
}
return ans;
}
void pls(int x0, int y0, int x1, int y1, int k) {
change(0, x0 + 1, y0 + 1, (x0 * y0) * k);
change(0, x0 + 1, y1 + 1, -(x0 * y1) * k);
change(0, x1 + 1, y0 + 1, -(x1 * y0) * k);
change(0, x1 + 1, y1 + 1, (x1 * y1) * k);
//-------
change(1, x0 + 1, y0 + 1, -(y0 * k));
change(1, x1 + 1, y0 + 1, (y0 * k));
change(1, x0 + 1, y1 + 1, (y1 * k));
change(1, x1 + 1, y1 + 1, -(y1 * k));
//-------
change(2, x0 + 1, y0 + 1, -(x0 * k));
change(2, x0 + 1, y1 + 1, (x0 * k));
change(2, x1 + 1, y0 + 1, (x1 * k));
change(2, x1 + 1, y1 + 1, -(x1 * k));
//-------
change(3, x0 + 1, y0 + 1, k);
change(3, x0 + 1, y1 + 1, -k);
change(3, x1 + 1, y0 + 1, -k);
change(3, x1 + 1, y1 + 1, k);
}
int sum(int x, int y) {
return qu(0, x, y) + (qu(1, x, y) * x) + (qu(2, x, y) * y) +
(qu(3, x, y) * x * y);
}
int query(int x0, int y0, int x1, int y1) {
int sum1 = sum(x0 - 1, y0 - 1);
int sum2 = sum(x0 - 1, y1);
int sum3 = sum(x1, y0 - 1);
int sum4 = sum(x1, y1);
return sum4 - sum2 - sum3 + sum1;
}
int main() {
#ifdef D
freopen("input", "r", stdin);
#endif
scanf("%c %d %d", &opt, &n, &m);
n++, m++;
memset(c, 0, sizeof(c));
while (scanf("%c", &opt) == 1)
if (opt == 'L' || opt == 'k') {
int x0, y0, x1, y1;
scanf("%d %d %d %d", &x0, &y0, &x1, &y1);
x0++;
x1++;
y0++;
y1++;
if (x0 > x1) {
std::swap(x0, x1);
std::swap(y0, y1);
}
if (opt == 'L') {
int v;
scanf("%d ", &v);
x0--, y0--;
pls(x0, y0, x1, y1, v);
}
if (opt == 'k') {
#ifdef D
for (int k = 0; k < 4; k++) {
for (int i = 1; i <= n; i++) {
for (int j = 1; j <= m; j++)
std::cout << c[k][i][j] << ' ';
std::cout << std::endl;
}
std::cout << std::endl << std::endl;
}
#endif
int ans = query(x0, y0, x1, y1);
printf("%d\n", ans);
}
}
}
[bzoj3132]上帝造题的七分钟——二维树状数组的更多相关文章
- 【bzoj3132】上帝造题的七分钟 二维树状数组区间修改区间查询
题目描述 “第一分钟,X说,要有矩阵,于是便有了一个里面写满了0的n×m矩阵. 第二分钟,L说,要能修改,于是便有了将左上角为(a,b),右下角为(c,d)的一个矩形区域内的全部数字加上一个值的操作. ...
- 【BZOJ3132】【TYVJ1716】上帝造题的七分钟 二维树状数组
题目大意 维护一个\(n\times m\)的矩阵,有两种操作: \(1~x_1~y_1~x_2~y_2~v\):把\((a,b),(c,d)\)为顶点的矩形区域内的所有数字加上\(v\). \(2~ ...
- tyvj P1716 - 上帝造题的七分钟 二维树状数组区间查询及修改 二维线段树
P1716 - 上帝造题的七分钟 From Riatre Normal (OI)总时限:50s 内存限制:128MB 代码长度限制:64KB 背景 Background 裸体就意味着 ...
- P4514 上帝造题的七分钟——二维树状数组
P4514 上帝造题的七分钟 求的是矩阵里所有数的和: 维护四个树状数组: #include<cstdio> #include<cstring> #include<alg ...
- BZOJ 3132: 上帝造题的七分钟( 二维BIT )
二维树状数组... 自己YY一下再推一下应该可以搞出来... --------------------------------------------------------------------- ...
- POJ 2029 Get Many Persimmon Trees (模板题)【二维树状数组】
<题目链接> 题目大意: 给你一个H*W的矩阵,再告诉你有n个坐标有点,问你一个w*h的小矩阵最多能够包括多少个点. 解题分析:二维树状数组模板题. #include <cstdio ...
- P4514 上帝造题的七分钟(二维树状数组)
P4514 上帝造题的七分钟 二维树状数组 差分维护区间加法,区间求和 #include<cstdio> int read(){ ,f=; ') f=f&&(c!='-') ...
- POJ 2155 Matrix (二维树状数组)
Matrix Time Limit: 3000MS Memory Limit: 65536K Total Submissions: 17224 Accepted: 6460 Descripti ...
- POJ_2155 Matrix 【二维树状数组】
一.题面 POJ2155 二.分析 楼教主出的题,是二维树状数组非常好的题,还结合了开关问题(开关变化的次数如果为偶数,状态不变,奇数状态相反). 题意就是给了一个二维的坐标平面,每个点初始值都是0, ...
随机推荐
- IDA动态调试SO文件
1. 所需工具 IDA Pro 6.6. 安卓SDK工具 2. 模拟器设置 将ida所在目录的dbgsrv文件夹内的android_server文件push到模拟器中. 设置777属性 启动调试服务器 ...
- 【廖雪峰老师python教程】——map/reduce
Map[单个操作对不同单一对象重复进行] map()函数接收两个参数,一个是函数,一个是Iterable,map将传入的函数依次作用到序列的每个元素,并把结果作为新的Iterator返回. 返回结果注 ...
- 「日常训练」「小专题·USACO」 Ski Course Design (1-4)
题目 以后补 分析 mmp这题把我写蠢哭了 我原来的思路是什么呢? 每轮找min/max,然后两个决策:升min/降max 像这样子dfs找最优,然后花式剪枝 但是一想不对啊,这才1-4,哪有那么复杂 ...
- JMeter学习笔记(九) 参数化4--User Variables
4.User Variables 用户参数 1)线程组右键添加 -> 前置处理器 -> 用户参数 2)配置用户参数 3)添加HTTP请求,引用用户参数,格式: ${} 4)配置线程数 5) ...
- pytest 测试报告
测试报告 运行测试用例后,为了保存结果,我们需要生成测试报告,同时可以把运行的测试报告发送相关人员查阅,这时需要安装一个插件(pytest-html) pytest-html插件安装 pip inst ...
- Web负载均衡技术
Web负载均衡(Load Balancing),简单地说就是给我们的服务器集群分配“工作任务”,而采用恰当的分配方式,对于保护处于后端的Web服务器来说,非常重要. 负载均衡的策略有很多,我们从简单的 ...
- Docker 安装Neo4j
拉取最新的neo4j镜像 docker pull neo4j 运行Neo4j 容器 docker run -it -d -p 7474:7474 -p 7687:7687 neo4j:latest 打 ...
- 用express搭建一个简单的博客系统
转自:https://blog.csdn.net/qq_29721837/article/details/62055603 Express 简介 Express 是一个简洁而灵活的 node.js W ...
- 【EasyNetQ】- 自动订阅者
从v0.7.1.30开始,EasyNetQ简单易用AutoSubscriber.你可以用它来轻松地扫描实现任何接口的类的特定组件IConsume<T>或IConsumeAsync<T ...
- struts2 下载文件
作者:禅楼望月 当下载的文件名字中不含有汉字,或者下载的文件不需要考虑用户的权限问题时.直接让超链接的href属性为所要下载的文件名即可.否则最好使用struts2的文件下载机制. 以下载图片为例 完 ...