题目链接: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 ;
}

最新文章

  1. em 换算
  2. MongoDB C API
  3. CLR via C#(06)- 构造器
  4. 2Sigma OA prepare: Longest Chain
  5. Servlet配置信息
  6. Chapter7: question 49 - 50
  7. python Quicksort demo
  8. 源码分析-mysql
  9. uploadify多图片上传实例
  10. RequireJS入门(三)转
  11. CentOS安装 pure-ftpd
  12. 释放SQL Server占用的内存
  13. 自己做的网页页面导航浏览JS/JQuery
  14. find the most comfortable road(并差集,找差值最小的权值)
  15. _foreach
  16. JavaScript之获取节点
  17. navigator.userAgent浏览器检测(前端基础系列)
  18. Apache Arrow
  19. 走进JDK(十二)------TreeMap
  20. 2-SET 前缀优化建图

热门文章

  1. python之路——25
  2. promise规范之部分总结
  3. 前端使用crypto.js进行加密
  4. 淘宝App直播宝贝数据采集
  5. sprinmvc与 Struct2框架的区别
  6. mysql索引总结(转)
  7. vue3.0中如何使用ueditor
  8. Python Queue(队列)
  9. xpath 选取指定文本内容可能是多种情况下的语法
  10. 转载:深入浅出Zookeeper(一) Zookeeper架构及FastLeaderElection机制