洛谷P3655 差分数组 树状数组
题目链接:https://www.luogu.org/problemnew/show/P3655
不一定对,仅供参考,不喜勿喷,不喜勿喷。
先copy洛谷P3368 【模板】树状数组 2 题解里面一位大佬Lyp10000对差分数组的解释:
来介绍一下差分
设数组a[]={,,,,},那么差分数组b[]={,,,-,}
也就是说b[i]=a[i]-a[i-];(a[]=;),那么a[i]=b[]+....+b[i];(这个很好证的)。
假如区间[,]都加上2的话
a数组变为a[]={,,,,},b数组变为b={,,,-,};
发现了没有,b数组只有b[]和b[]变了,因为区间[,]是同时加上2的,所以在区间内b[i]-b[i-]是不变的.
所以对区间[x,y]进行修改,只用修改b[x]与b[y+]:
b[x]=b[x]+k;b[y+]=b[y+]-k;
看题目很容易想到差分数组,但是现在不是直接求出差分数组的和,而是要对求出的差分数组的每一个元素判断正负之后,然后再通过题目中给的那个公式进行加工,最后求出加工之后数组的和,在这里我们吧加工之后的数组用树状数组来存,这样容易修改和求和,我们分别用数组a,b,c来代表原始数据,原始数据的差分数组,差分数组对应的树状数组。
因为树状数组的值取决于差分数组的值,根据差分数组的性质,当我们在修改数组a(原始数据)的某个区间,即同时让a数组上区间[L,R]之间的数增加value,事实上在差分数组里面(b数组)只改变了b[L],和b[R+1](如果R+1不越界的话),其中b[L]增加了value,b[R+1]减少了value。根据这个性质,我们发现,因为差分数组每次只改变两个值(b[L],b[R+1]),那么其实对应的树状数组也只需要执行两个单点修改的操作,即add(L,?),add(R+1,?),这里面的两个问号就代表着我们要在树状数组里面单点修改时改变的差值。
那么这个改变的差值是很好求的,只要求出未改变之前的值,和改变之后对应的值,用后者减前者就是改变的差值。也就是这个操作:
LL cal(LL x,LL value){//cal函数是计算当差分数组的b[x]加上value时,对应的树状数组应该更改的值
LL A,B;//a代表未改变时的值,b表示改变之后的值
//b-a就是他们之间的差值,也就是改变的差值
if(b[x]>)
A=-S*abs(b[x]);
else
A=T*abs(b[x]);
if(b[x]+value>)
B=-S*abs(b[x]+value);
else
B=T*abs(b[x]+value);
return B-A;
/*if(b[x]>0){
if(b[x]+value>0)
return -S*(abs(b[x]+value))-(-S*abs(b[x]));
else
return T*(abs(b[x]+value))-(-S*abs(b[x]));
}else{
if(b[x]+value<0)
return T*(abs(b[x]+value))-(T*abs(b[x]));
else
return -S*(abs(b[x]+value))-(T*abs(b[x]));
}*/
}
代码:
#include<iostream>
#include<cstring>
#include<algorithm>
#include<queue>
#include<map>
#include<stack>
#include<cmath>
#include<vector>
#include<set>
#include<cstdio>
#include<string>
#include<deque>
using namespace std;
typedef long long LL;
#define eps 1e-8
#define INF 0x3f3f3f3f
#define maxn 200005
LL N,Q,S,T;
LL a[maxn],b[maxn],c[maxn];
LL lowbit(LL x){
return x&(-x);
}
LL sum(LL x){
LL ans=;
while(x){
ans+=c[x];
x-=lowbit(x);
}
return ans;
}
void add(LL x,LL value){
while(x<maxn){
c[x]+=value;
x+=lowbit(x);
}
}
LL cal(LL x,LL value){//cal函数是计算当差分数组的b[x]加上value时,对应的树状数组应该更改的值 LL A,B;//a代表未改变时的值,b表示改变之后的值
//b-a就是他们之间的差值,也就是改变的差值
if(b[x]>)
A=-S*abs(b[x]);
else
A=T*abs(b[x]); if(b[x]+value>)
B=-S*abs(b[x]+value);
else
B=T*abs(b[x]+value);
return B-A; /*if(b[x]>0){
if(b[x]+value>0)
return -S*(abs(b[x]+value))-(-S*abs(b[x]));
else
return T*(abs(b[x]+value))-(-S*abs(b[x]));
}else{
if(b[x]+value<0)
return T*(abs(b[x]+value))-(T*abs(b[x]));
else
return -S*(abs(b[x]+value))-(T*abs(b[x]));
}*/
}
int main()
{
while(scanf("%lld%lld%lld%lld",&N,&Q,&S,&T)!=EOF){
memset(c,,sizeof(c));//树状数组清零
scanf("%lld",&a[]);
b[]=;
for(int i=;i<=N;i++){
scanf("%lld",&a[i]);
b[i]=a[i]-a[i-];//b数组是差分数组,即b[i]=a[i]-a[i-1](定义)
if(b[i]<)//先把一开始的差分数组对应的树状数组求出来
add(i,T*abs(b[i]));
else
add(i,-S*abs(b[i]));
}
LL x,y,value;
while(Q--){
scanf("%lld%lld%lld",&x,&y,&value);
add(x,cal(x,value));//cal函数得到c[x]位置处的改变值,然后对树状数组进行单点修改
if(y+<=N){//判断是否越界
add(y+,cal(y+,-value));//和上面一样
}
b[x]+=value;//前面的b数组不可以改变,在这里改变他的值,因为我们每次都要和前一次值只比较
b[y+]-=value;
printf("%lld\n",sum(N));//输出差分数组的总和
}
}
return ;
}
洛谷P3655 差分数组 树状数组的更多相关文章
- 【洛谷 p3374】模板-树状数组 1(数据结构)
题目:已知一个数列,你需要进行下面两种操作:1.将某一个数加上x:2.求出某区间每一个数的和. 解法:树状数组求前缀和. #include<cstdio> #include<cstd ...
- 【洛谷 p3368】模板-树状数组 2(数据结构)
题目:已知一个数列,你需要进行下面两种操作:1.将某区间每一个数数加上x:2.求出某一个数的和. 解法:树状数组+前缀和优化.数组中每位存和前一位的数的差,这样区间修改只用改两位,单点询问就是求前缀和 ...
- 洛谷 P1908 逆序对(树状数组解法)
归并排序解法:https://www.cnblogs.com/lipeiyi520/p/10356882.html 题目描述 猫猫TOM和小老鼠JERRY最近又较量上了,但是毕竟都是成年人,他们已经不 ...
- AcWing:246. 区间最大公约数(线段树 + 增量数组(树状数组) + 差分序列)
给定一个长度为N的数列A,以及M条指令,每条指令可能是以下两种之一: 1.“C l r d”,表示把 A[l],A[l+1],…,A[r] 都加上 d. 2.“Q l r”,表示询问 A[l],A[l ...
- BZOJ 1935: [Shoi2007]Tree 园丁的烦恼( 差分 + 离散化 + 树状数组 )
假如矩阵范围小一点就可以直接用二维树状数组维护. 这道题, 差分答案, 然后一维排序, 另一维离散化然后树状数组维护就OK了. ----------------------------------- ...
- hdu1081 DP类最大子段和(二维压缩+前缀和数组/树状数组计数)
题意:给出一个 n * n 的数字矩阵,问最大子矩阵和是多少. 由于和最长子段和问题类似,一开始想到的就是 DP ,一开始我准备用两个循环进行 DP ,对于每一个 (i,j) ,考察(i - 1,j) ...
- BZOJ 2754 SCOI 2012 喵星球上的点名 后缀数组 树状数组
2754: [SCOI2012]喵星球上的点名 Time Limit: 20 Sec Memory Limit: 128 MBSubmit: 2068 Solved: 907[Submit][St ...
- Codeforces Good Bye 2015 D. New Year and Ancient Prophecy 后缀数组 树状数组 dp
D. New Year and Ancient Prophecy 题目连接: http://www.codeforces.com/contest/611/problem/C Description L ...
- bzoj 3473 字符串 - 后缀数组 - 树状数组
题目传送门 传送门 题目大意 给定n个字符串,询问每个字符串有多少子串(不包括空串)是所有n个字符串中至少k个字符串的子串 先用奇怪的字符把所有字符串连接起来. 建后缀树,数每个节点的子树内包含多少属 ...
随机推荐
- nginx1.14.0版本https加密配置
修改host文件,为最后访问域名准备 C:\Windows\System32\drivers\etc host文件目录192.168.10.140 www.joyce.com 在最后添加这个自定义域名 ...
- [转][ActiveMQ]Apache.NMS.ActiveMQ 用法
下载 C# 组件:http://archive.apache.org/dist/activemq/apache-nms/1.7.0/ 使用说明:https://www.cnblogs.com/cjm1 ...
- [UE4]让Spline具象化
接上一个实例 一.在TestSpline蓝图,切换到蓝图构造函数Constrction Script事件中,添加如下代码: 二.别忘记个Add Spline Mesh Component设置Stati ...
- webservice之jax-rs实现方式
1.什么叫restful风格 restful是一组架构约束条件和原则,满足这些约束条件和原则的应用程序即是restful风格. 2.jax-rs实现步骤 1.创建一个简单应用(略) 2.添加依赖jar ...
- 检查邮箱IP是否在国际反垃圾邮件组织的黑名单中
有时候邮件发不出去,很有可能就是邮件服务器的IP被国际上一些反垃圾组织列入黑名单了,这时你可以通过返回的邮件判断是否进入黑名单,或者通过以下查询地址看是否被列入,然后一个个申请移除: http://m ...
- oracle Extract 函数
//oracle中extract()函数从oracle 9i中引入,用于从一个date或者interval类型中截取到特定的部分 //语法如下: EXTRACT ( { Y ...
- Windows下安装ZooKeeper
Windows下安装ZooKeeper 一.简介 ZooKeeper是一个分布式的,开放源码的分布式应用程序协调服务,是Google的Chubby一个开源的实现,是Hadoop和Hbase的重要组 ...
- 记一次bond引起的网络故障
本案中3个关键服务器 物理服务器:192.168.6.63,简称P,(Physical server) KVM-VM:192.168.6.150,是物理服务器P上的一个KVM虚机,简称VM NAS:外 ...
- python 中list的深拷贝和浅拷贝
注意:这里提到是list的隐式转换例如 a=[1,2,3]def test(n):n[2] = n[2]*2c=a此时是浅拷贝,当调用test(c)时,可以看到a也发生了变化c=[1,4,3]a=[1 ...
- 6.1 集合和映射--集合Set->底层基于二叉搜索树实现
前言:在第5章的系列学习中,已经实现了关于二叉搜索树的相关操作,详情查看第5章即可.在本节中着重学习使用底层是我们已经封装好的二叉搜索树相关操作来实现一个基本的集合(set)这种数据结构.集合set的 ...