luo3372线段树模板的分块做法
题目大意
请你维护一个有n个元素的整数序列,要求支持区间查询&区间修改
对于100%的数据,\(1<=n<=10^5\)
分析
正常做法是线段树维护区间修改、区间查询,今天我要讲的是一种暴力做法:分块
分块的思想并不复杂,分块把一个长度为n的区间分成num段,操作时如果是整段用标记修改,不是整段的部分暴力修改
分析时间复杂度:在这题中,每段的标记修改是\(O(1)\)的,最多有num段,整段标记修改所用时间是\(O(num)\)的;不是整段的部分最多有\(O(n/num)\)个,暴力修改所用的时间是\(O(n/num)\)的;所以总时间是\(O(num+n/num)\)。
根据基本不等式,num取\(\sqrt n\)时该式有最小值;所以num取\(\sqrt n\)。
实现
分块的思想并不复杂,时间复杂度也不玄学,但是实现起来并不方便(可能是我弱)
分块的修改/查询都分为2部分:
- 整块的修改
- “零头”的修改
整块的修改是否简便:add[i] += val;
“零头”的修改直接修改,同时不要忘了维护所在块的信息:
a[i] += val;
sum[id[i]] += val;
修改就愉快地解决了,查询和修改差不多。
完整代码
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
using namespace std;
const int maxn = 100007;
int n, m, num, id[maxn];
long long sum[1000], add[1000], a[maxn];
int main(){
scanf("%d%d", &n, &m);
num = sqrt(n);
for (int i = 1; i <= n; ++i){
scanf("%lld", &a[i]);
id[i] = (i-1) / num;
sum[id[i]] += a[i];
}
while (m--){
int d; scanf("%d", &d);
if (d==1){
int x, y, C; scanf("%d%d%d", &x, &y, &C);
int Leftid = (x-1) / num + 1;
int Rightid = (y-1) / num - 1;
long long res = 0;
for (int i = Leftid; i <= Rightid; ++i)
add[i] += C;
for (int i = x; i <= Leftid * num; ++i)
a[i] += C, sum[id[i]] += C;
for (int i = (Rightid+1)*num+1; i <= y; ++i)
a[i] += C, sum[id[i]] += C;
}else{
int x, y; scanf("%d%d", &x, &y);
int Leftid = (x-1) / num + 1;
int Rightid = (y-1) / num - 1;
if (id[x] == id[y]){
long long res = 0;
for (int i = x; i <= y; ++i)
res += a[i] + add[id[i]];
printf("%lld\n", res);
continue;
}
long long res = 0;
for (int i = Leftid; i <= Rightid; ++i)
res += sum[i] + add[i] * num;
for (int i = x; i <= Leftid * num; ++i)
res += a[i] + add[Leftid-1];
for (int i = (Rightid+1)*num+1; i <= y; ++i)
res += a[i] + add[Rightid+1];
printf("%lld\n", res);
}
}
return 0;
}
luo3372线段树模板的分块做法的更多相关文章
- 洛谷P4344 脑洞治疗仪 [SHOI2015] 线段树+二分答案/分块
!!!一道巨恶心的数据结构题,做完当场爆炸:) 首先,如果你用位运算的时候不小心<<打成>>了,你就可以像我一样陷入疯狂的死循环改半个小时 然后,如果你改出来之后忘记把陷入死循 ...
- [AHOI 2009] 维护序列(线段树模板题)
1798: [Ahoi2009]Seq 维护序列seq Time Limit: 30 Sec Memory Limit: 64 MB Description 老师交给小可可一个维护数列的任务,现在小 ...
- hdu1754 I hate it线段树模板 区间最值查询
题目链接:这道题是线段树,树状数组最基础的问题 两种分类方式:按照更新对象和查询对象 单点更新,区间查询; 区间更新,单点查询; 按照整体维护的对象: 维护前缀和; 维护区间最值. 线段树模板代码 # ...
- P3373 线段树模板
好,这是一个线段树模板. #include <cstdio> using namespace std; ; long long int sum[N],tag1[N],tag2[N],mo; ...
- 线段树模板hdu 1754:I Hate It
I Hate It Time Limit: 9000/3000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total S ...
- UESTC - 1057 秋实大哥与花 线段树模板题
http://acm.uestc.edu.cn/#/problem/show/1057 题意:给你n个数,q次操作,每次在l,r上加上x并输出此区间的sum 题解:线段树模板, #define _CR ...
- POJ 3468 A Simple Problem with Integers(线段树模板之区间增减更新 区间求和查询)
A Simple Problem with Integers Time Limit: 5000MS Memory Limit: 131072K Total Submissions: 140120 ...
- hdu 4819 二维线段树模板
/* HDU 4819 Mosaic 题意:查询某个矩形内的最大最小值, 修改矩形内某点的值为该矩形(Mi+MA)/2; 二维线段树模板: 区间最值,单点更新. */ #include<bits ...
- POJ3468:A Simple Problem with Integers(线段树模板)
A Simple Problem with Integers Time Limit: 5000MS Memory Limit: 131072K Total Submissions: 149972 ...
随机推荐
- Gym - 101350A Sherlock Bones(思维)
The great dog detective Sherlock Bones is on the verge of a new discovery. But for this problem, he ...
- mysql 单列无重复
ALTER TABLE jeesite.bb_bill ADD UNIQUE (object_id);
- Client-Side Template Injection with AngularJS
<html> <head> <meta charset="utf-8"> <script src="https://cdn.bo ...
- 第三节: EF调用普通SQL语句的两类封装(ExecuteSqlCommand和SqlQuery )
一. 前言 在前面的两个章节中,我们分别详细介绍了EF的增删改的两种方式(方法和状态)和EF查询的两种方式( Lambda和Linq ),进行到这里,可以说对于EF,已经入门了,本来应该继续往下进行E ...
- cookie小栗子-实现简单的身份验证
关于Cookie Cookie是一种能够让网站Web服务器把少量数据储存到客户端的硬盘或内存里,或是从客户端的硬盘里读取数据的一种技术. 用来保存客户浏览器请求服务器页面的请求信息,可以在HTTP返回 ...
- SpringBoot系列: 使用 Swagger 生成 API 文档
SpringBoot非常适合开发 Restful API程序, 我们都知道为API文档非常重要, 但要维护好难度也很大, 原因有: 1. API文档如何能被方便地找到? 以文件的形式编写API文档都有 ...
- [物理学与PDEs]第2章习题13 将 $p$ - 方程组化为守恒律形式的一阶拟线性对称双曲组
试引进新的未知函数, 将 $p$ - 方程组 $$\beex \bea \cfrac{\p \tau}{\p t}-\cfrac{\p u}{\p x}&=0,\\ \cfrac{\p u}{ ...
- 对评分矩阵进行分解,SVD与LSI
摘自 推荐系统 https://www.cnblogs.com/lzllovesyl/p/5243370.html 一.SVD奇异值分解 1.SVD简介 SVD(singular value deco ...
- String.intern() 方法__jdk1.6与jdk1.7/jdk1.8的不同
1.为什么要使用intern()方法 intern方法设计的初衷是为了重用string对象,节省内存 用代码实例验证下 public class StringInternTest { static f ...
- rxjs
流就是一个事件 或者执行的某些操作