- 1、本文档共6页,可阅读全部内容。
- 2、原创力文档(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。
- 3、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载。
- 4、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
- 5、该文档为VIP文档,如果想要下载,成为VIP会员后,下载免费。
- 6、成为VIP后,下载本文档将扣除1次下载权益。下载后,不支持退款、换文档。如有疑问请联系我们。
- 7、成为VIP后,您将拥有八大权益,权益包括:VIP文档下载权益、阅读免打扰、文档格式转换、高级专利检索、专属身份标志、高级客服、多端互通、版权登记。
- 8、VIP文档为合作方或网友上传,每下载1次, 网站将根据用户上传文档的质量评分、类型等,对文档贡献者给予高额补贴、流量扶持。如果你也想贡献VIP文档。上传文档
查看更多
树状数组维区间和的模型及其拓广的简单总结
树状数组维护区间和的模型及其拓广的简单总结
by wyl8899
树状数组的基本知识已经被讲到烂了,我就不多说了,下面直接给出基本操作的代码。
假定原数组为a[1..n],树状数组b[1..n],考虑灵活性的需要,代码使用int *a传数组。
#define lowbit(x) ((x)(-(x)))
int sum(int *a,int x){
int s=0;
for(;x;x-=lowbit(x))s+=a[x];
return s;
}
void update(int *a,int x,int w){
for(;x=n;x+=lowbit(x))a[x]+=w;
}
sum(x)返回原数组[1,x]的区间和,update(x,w)将原数组下标为x的数加上w。
这两个函数使用O(操作数*logn)的时间和O(n)的空间完成单点加减,区间求和的功能。
接下来做一些升级,让树状数组完成区间加减,单点查询的功能。
直接做的话很困难,需要对问题做一些转化。
考虑将原数组差分,即令d[i]=a[i]-a[i-1],特别地,d[1]=a[1]。
此时a[i]=d[1]+..+d[i],所以单点查询a[i]实际上就是在求d数组的[1..i]区间和。
而区间[l,r]整体加上k的操作,可以简单地使用d[l]+=k和d[r+1]-=k来完成。
于是,我们用树状数组来维护d[],就可以解决问题了。
下面再升级一次,完成区间加减,区间求和的功能。
仍然沿用d数组,考虑a数组[1,x]区间和的计算。d[1]被累加了x次,d[2]被累加了x-1次,...,d[x]被累加了1次。
因此得到
sigma(a[i])
=sigma{d[i]*(x-i+1)}
=sigma{ d[i]*(x+1) - d[i]*i }
=(x+1)*sigma(d[i])-sigma(d[i]*i)
所以我们再用树状数组维护一个数组d2[i]=d[i]*i,即可完成任务。
POJ 3468就是这个功能的裸题,下面给出代码。
[请注意我们上面的讨论都假定了a[]初始全是0。如果不是这样呢?下面的程序里给出了一个相对简便的处理办法。]
// POJ 3468 Using BIT
#include cstdio
const int maxn=100010;
__int64 a[maxn],b[maxn],c[maxn];
int n,m;
inline int lowbit(const int x){
return x(-x);
}
__int64 query(__int64 *a,int x){
__int64 sum=0;
while(x){sum+=a[x];x-=lowbit(x);}
return sum;
}
void update(__int64 *a,int x,__int64 w){
while(x=n){a[x]+=w;x+=lowbit(x);}
}
int main(){
int l,r,i;
__int64 ans,w;
char ch;
scanf(%d%d,n,m);
a[0]=0;
for(i=1;i=n;++i){
scanf(%I64d,a[i]);
a[i]+=a[i-1];
}
while(m--){
scanf(%c,ch);
while(ch!=Q ch!=C)scanf(%c,ch);
if(ch==Q){
scanf(%d%d,l,r);
ans=a[r]-a[l-1]+(r+1)*query(b,r)-l*query(b,l-1)-query(c,r)+query(c,l-1);
printf(%I64d\n,ans);
}else{
scanf(%d%d%I64d,l,r,w);
update(b,l,w);
update(b,r+1,-w);
update(c,l,w*l);
update(c,r+1,-(r+1)*w);
}
}
return 0;
}
[当a[]初始不全0的时候,我们就只维护后来加上去的部分,查询区间和的时候再补上初始的时候这一段的区间和就可以了。]
======================一维到二维的分割线=========================
接下来到二维树状数组。
先看看sum和update变成什么样子了吧。
inline int gs(int a[maxn][m
您可能关注的文档
- 机动车辆进管理作业规程.doc
- 机械原理课设计牛头刨床的VB.doc
- 机械CAD应用实验指导书.doc
- 机械原理陈柜_机械基础示教陈列柜_机械陈列柜大全_-上海中义.doc
- 机械原理课设计偏置直动滚子从动杆盘型凸轮机构.doc
- 朝阳高二会模拟1212.doc
- 机械维修工论试题.doc
- 机械工程控基础复习资料.doc
- 机械振动填问答参考(力学).doc
- 机械基础力资料.doc
- 数据仓库:Redshift:Redshift与BI工具集成.docx
- 数据仓库:Redshift:数据仓库原理与设计.docx
- 数据仓库:Snowflake:数据仓库成本控制与Snowflake定价策略.docx
- 大数据基础:大数据概述:大数据处理框架MapReduce.docx
- 实时计算:GoogleDataflow服务架构解析.docx
- 分布式存储系统:HDFS与MapReduce集成教程.docx
- 实时计算:Azure Stream Analytics:数据流窗口与聚合操作.docx
- 实时计算:Kafka Streams:Kafka Streams架构与原理.docx
- 实时计算:Kafka Streams:Kafka Streams连接器开发与使用.docx
- 数据仓库:BigQuery:BigQuery数据分区与索引优化.docx
文档评论(0)