【转载】区间信息的维护与查询(一)——二叉索引树(Fenwick树、树状数组)

时间:2022-09-17 15:38:29

在网上找到一篇非常不错的树状数组的博客,拿来转载,原文地址

树状数组

最新看了一下区间的查询与修改的知识,最主要看到的是树状数组(BIT),以前感觉好高大上的东西,其实也不过就这么简单而已。

我们有一个动态连续和查询问题:给定一个n个元素的数组A1,A2,A3,…An,你的任务是设计一个数据结构,使得其支持以下两个操作:

1:Add(x,d)操作:让Ax增加d;

2:Query(L,R)操作:计算AL+AL+1+⋯+AR。

第一种思路就是循环累加,这样每次的时间复杂度都是Θ(n)级别的。这样在数据很大的情况之下,是一定会效率很低的,所以我们引进了二叉索引树(也就是树状数组)这种比较高级的数据结构,说它高级,也高不到那里去,也就是比原先我们学过的数据结构难一些就是了,这种数据结构在NOI中是经常使用的,但是NOIP一般不考,我也是一个NOIPer,水平也是达不到NOI的,只是我感觉这个东西挺好玩,也挺实用的,所以就学了学,发一篇博客。

在讲BIT之前,我们来先了解一个函数:对于任意正整数x,我们定义lowbit(x)为x的二进制中最右边的1所对应的值,比如,5的二进制是101,那么lowbit(5)= 1;4的二进制是100,那么lowbit(4) = 4;在程序实现中,lowbit代码如下

(此处对原文中的代码有修改)

int lowbit (int x)
{
return x & (-x);
}

这里用到的是按位运算,请读者自己去查阅关于这点的资料。但为什么呢?计算机里面的整数采用补码表示,-x实际上是x在二进制中按位取反,末位+1后的结果,二者按位取“与”之后,前面全部变成0,之后的lowbit保持不变;

接下来给大家附上一张BIT 的图,其实也不是很难懂,但是我想要的图我找不到了,所以就附一个别的图吧,希望大家能尽量去看,在下面我会给大家解释其中C数组的含义

【转载】区间信息的维护与查询(一)——二叉索引树(Fenwick树、树状数组)

其中我们可以看到C[i]是有分层问题的,那么到底是怎么分层的呢,实际上就是根据lowbit(i)的值来分的层。

在这里我们可以看到BIT是有连线的,但实际上这些连线在计算机中并不存在,只是为了读者好理解才加上的。其实BIT本身就是一棵二叉树(具体请翻阅前面BIT的定义),那么这棵树上面就会有父亲节点和左右儿子节点(实际上在图中没有看到右孩子与父亲节点的连线,我们可以想象一下)

关于在BIT上节点的父子关系,我们是这样定义的:

对于节点i,如果它是左子节点,那么它的父节点的编号为i+lowbit(i);如果它是右子节点,那么它的父节点编号为i+lowbit(i)。大家可以自己证明一下这个东西。搞清楚BIT 结构之后,我们来讲一下这个C数组是干嘛的。

C数组实际上只是个辅助数组,其中C[i]=A[i-lowbit(i)+1]+A[i-lowbit(i)+2]+……+A[i];可以看出,C数组就是用来辅助计算前缀和S[i]的;

那么我们有了C数组之后,如何计算S[i]呢?顺着i节点往左走,边走边往上爬(这里请注意,不一定沿着BIT中的边爬),把沿途的C[i]累加起来就好了(请读者注意,这里沿途经过的C[i]毫无遗漏和重复的走完了A[i]),那么下面我给大家附上这个求前缀和操作的代码(这里我只会给出核心代码,因为全部代码给出真的就是没必要):

int Query(int x){
int ret = 0;
while(x > 0){
ret += C[x];
x-=lowbit(x);
}
return ret;
}

下面来讲一下修改问题,因为BIT是一棵树,而且根据前面的C[i]的定义,我们可以知道,当某个A[i]改变时,有一些C[i]也会改变,那么需要更改那些C数组中的元素呢?从C[i]开始往右走,边走边“往上爬”(同上,不一定要沿着图中的边爬),沿途修改经过所有节点对应的C[i]值即可。下面附上更改元素的代码(同样只给出核心代码):

void add(int x,int d){
while(x <= n){
C[x] += d;
x += lowbit(x);
}
}

这两个操作的时间复杂度都是Θ(log2n)预处理方法是将A和C数组清空,再执行n次add操作,总时间复杂度为Θ(nlog2n);

BIT的推广:二位BIT:

一维BIT很容易推广到二维,二维BIT如下所示:

C[x][y] = sum(A[i][j]);

二维BIT的应用:

一个由数字构成的大矩阵,支持以下操作:

1.给矩阵中某个元素加上一个整数d(正负都可以);

2.查询某个子矩阵中所有数字的和

并且对每个查询操作输出结果.

转载于2015年8月29日。

【转载】区间信息的维护与查询(一)——二叉索引树(Fenwick树、树状数组)的更多相关文章

  1. 树状数组(二叉索引树 BIT Fenwick树) &ast;【一维基础模板】(查询区间和&plus;修改更新)

    刘汝佳:<训练指南>Page(194) #include <stdio.h> #include <string.h> #include <stdlib.h&g ...

  2. hdu 1166 敌兵布阵——(区间和)树状数组&sol;线段树

    pid=1166">here:http://acm.hdu.edu.cn/showproblem.php?pid=1166 Input 第一行一个整数T.表示有T组数据. 每组数据第一 ...

  3. 牛客练习赛52 B题【树状数组维护区间和&lbrace;查询区间和,如果区间元素重复出现则计数一次&rcub;】补题ing

    [题目] 查询区间和,如果区间元素重复出现则计数一次. 链接:https://ac.nowcoder.com/acm/contest/1084/B [题解] 将询问按r排序,维护每个数最后出现的位置, ...

  4. BZOJ 1018&colon; &lbrack;SHOI2008&rsqb;堵塞的交通traffic &lbrack;线段树 区间信息&rsqb;

    1018: [SHOI2008]堵塞的交通traffic Time Limit: 3 Sec  Memory Limit: 162 MBSubmit: 3064  Solved: 1027[Submi ...

  5. nyoj 123 士兵杀敌(四) 树状数组【单点查询&plus;区间修改】

    士兵杀敌(四) 时间限制:2000 ms  |  内存限制:65535 KB 难度:5   描述 南将军麾下有百万精兵,现已知共有M个士兵,编号为1~M,每次有任务的时候,总会有一批编号连在一起人请战 ...

  6. 线段树 区间开方区间求和 &amp&semi; 区间赋值、加、查询

    本文同步发表于 https://www.zybuluo.com/Gary-Ying/note/1288518 线段树的小应用 -- 维护区间开方区间求和 题目传送门 约定: sum(i,j) 表示区间 ...

  7. Libre OJ 130、131、132 (树状数组 单点修改、区间查询 -&gt&semi; 区间修改,单点查询 -&gt&semi; 区间修改,区间查询)

    这三题均可以用树状数组.分块或线段树来做 #130. 树状数组 1 :单点修改,区间查询 题目链接:https://loj.ac/problem/130 题目描述 这是一道模板题. 给定数列 a[1] ...

  8. Codeforces Round &num;425 &lpar;Div&period; 2&rpar; D 树链剖分 &plus; 树状数组维护区间

    一看就知道 可以LCA判断做 也可以树链剖分拿头暴力 然而快速读入和线段树维护区间会T70 于是只能LCA? 线段树的常数不小 于是需要另外一种办法来进行区间加减和查询区间和 就是使用树状数组 这个题 ...

  9. 牛客网 暑期ACM多校训练营(第二场)J&period;farm-STL&lpar;vector&rpar;&plus;二维树状数组区间更新、单点查询 or 大暴力?

    开心.jpg J.farm 先解释一下题意,题意就是一个n*m的矩形区域,每个点代表一个植物,然后不同的植物对应不同的适合的肥料k,如果植物被撒上不适合的肥料就会死掉.然后题目将每个点适合的肥料种类( ...

随机推荐

  1. 国内git项目托管平台

    以前一直使用github托管项目,最近换了阿里云的vps,连接github出奇的慢,找了一下国内的代码托管平台. 有几个都不错,我刚好有csdn的账号,就试了一下csdn的托管平台,创建一个项目,发现 ...

  2. autoresizingMask的用法

    UIViewAutoresizingNone = , UIViewAutoresizingFlexibleLeftMargin = << , UIViewAutoresizingFlexi ...

  3. URAL1133&period; Fibonacci Sequence(二分)

    1133 刚开始还用记忆化推了下公式 后来发现数是非常大的 二分 然后就是精度错误 中间值会很大 乱七八糟的改 #include <iostream> #include<cstdio ...

  4. fork 函数的一点学习

    昨天某位少年问了我一个问题,#include<stdio.h> int main() { fork(); fork(); fork(); printf("hello " ...

  5. Python3 官方文档翻译 - 4&period;7 函数定义

    4.7.1 默认函数定义 最常用的就是为一个或多个参数设定默认值,这让函数可以用比定义时更少的参数来调用,例如: def ask_ok(prompt, retries=4, complaint='Ye ...

  6. CodeForces 28D Don&amp&semi;&num;39&semi;t fear&comma; DravDe is kind dp

    主题链接:点击打开链接 为了让球队后,删除是合法的.也就是说,对于每一个车辆, l+r+c 一样,按l+r+c分类. 然后dp一下. #include <cstdio> #include ...

  7. 在Javascript中数组对象&lpar;json&rpar;里元素相同的操作

    1.数组对象元素相同,分组显示   let arry = [ { expensedate: '2018/09/29', amount: 1, type: '交通费' }, { expensedate: ...

  8. 背水一战 Windows 10 &lpar;110&rpar; - 通知(Tile)&colon; secondary tile 模板之基础&comma; secondary tile 模板之文本

    [源码下载] 背水一战 Windows 10 (110) - 通知(Tile): secondary tile 模板之基础, secondary tile 模板之文本 作者:webabcd 介绍背水一 ...

  9. KazaQ&&num;39&semi;s Socks (找规律)

    #include<iostream> using namespace std; #define ll long long ll n, m; ll t; int main(){ while ...

  10. vue笔记 - 生命周期第二次学习与理解

    对于刚接触vue一两个月.才仅仅独立做过一两个vue项目的小白来说,以前一直自我感觉自己知道vue的生命周期, 直到前两天去面试,面试官让我说一下vue的生命周期... 其实我的心中是有那张图的,但是 ...