树状数组(Binary Indexed Tree)
树状数组(Binary Indexed Tree,BIT) 是能够完成下述操作的数据结构。
给一个初始值全为 0 的数列 a1, a2, ..., an
(1)给定 i,计算 a1+a2+...+ai
(2)给定 i 和 x,执行 ai += x
1.基于线段树的实现
如果使用线段树,只需要做少许修改就可以实现这两个功能。线段树的每个节点上维护的是对应区间的和。
接下来看如何计算从 s 到 t 的和(as + as+1 + ... + at)。在基于线段树的实现这个和是可以直接求得的。
但是如果我们能够计算(从 1 到 t 的和) - (从 1 到 s - 1 的和),同样可以得到 s 到 t 的和。也就是说,只要对于任意 i,我们都能计算出 1 到 i 的部分和就可以了。
可以发现,线段树上的每个节点的右儿子的值都不需要了(在计算时如果要使用这个点的值,那么它的左边的兄弟的值也一定会用到,这个时候只需要使用它们的父亲的值就可以了)。
基于上面的思路得到的数据结构就是 BIT。比起线段树,BIT实现起来更方便,速度也更快。
2. BIT的结构
BIT 使用数组维护下图所示的部分和
也就是把线段树中不需要的节点去掉之后,再把剩下的节点对应到数组中。对比每个节点对应的区间的长度和节点编号的二进制表示。以 1 结尾的 1, 3, 5, 7 的长度是1, 最后有 1 个 0 的2, 6 的长度是2,最后有两个 0 的 4 的长度是 4 .... 这样,编号的二进制表示就能够和区间非常容易地对应起来。利用这个性质,BIT 可以通过非常简单的位运算实现。
3. BIT的求和
计算前 i 项的和需要从 i 开始,不断把当前位置 i 的值加入到结果中, 并从 i 中减去 i 的二进制最低非 0 位对应的幂,直到 i 变为 0 为止。i 的二进制的最后一个 1 可以通过 i & - i 得到。
4.BIT的值的更新
使第 i 项的值增加 x 需要从 i 开始,不断把当前位置 i 的值增加 x, 并把 i 的二进制最低非 0 位对应的幂加到 i 上。
5. BIT的复杂度
总共需要对O(log n)个值进行操作,所以复杂度是O(log n)。
6. BIT的实现
提一下 i -= i & -i 可以写为 i = i &(i - 1)
// [1, n]
int bit[MAX_N + ], n; int sum(int i) {
int s = ;
while (i > ) {
s += bit[i];
i -= i & -i;
}
return s;
} void add(int i, int x) {
while (i <= n) {
bit[i] += x;
i += i & -i;
}
}
//本来想把代码补完整的。。参考线段树就行,但是后面会有综述我就不补了
7. 二维BIT
BIT 可以方便地扩展到二维的情况。对于 W * H 的二维 BIT 只需建立 H 个大小为 x 轴方向元素个数 W 的 BIT,然后把这些 BIT 通过 y 轴方向的 BIT 管理起来就可以了。也就是说,y 轴方向的 BIT 的每个元素不是整数,而是一个 x 轴方向的 BIT。这样所有的操作的复杂度都是 O(log W * log H)。用同样的方法可扩展到更高的维度。
// emm
8. 需要运用 BIT 的问题
冒泡排序的复杂度是 O(n2),所以无法模拟过程。选用适当的数据结构可以解决这个问题。
所求的交换次数等价于 满足 i < j , ai > aj 的 i 的个数,(这种数对的个数叫做逆序数)。而对于每一个 j,如果能够快速求出满足 i < j, ai > aj 的 i 的个数,那么问题就解决了。我们构建一个值的范围是 1 ~ n 的 BIT,按照 j = 0, 1, 2, ..., n-1 的顺序进行如下操作。
(1)把 j - (BIT查询得到的前 aj 项的和)加到答案中
(2)把 BIT 中 aj 位置上的值加 1
对于每一个 j, (BIT查询得到的前 aj 项的和)就是满足i < j , ai > aj 的 i 的个数。因此把这个值从 j 中减去之后,得到的就是满足 i < j , ai > aj 的 i 的个数。由于对于每一个 j 的复杂度是O(log n),所以整个算法的复杂度是O(n log n)。
// 但实际上实现计算逆序数更简单的算法是通过分治思想利用归并排序计算
typedef long long ll;
int n, a[MAX_N];
//省略BIT部分代码
void solve() {
ll ans = ;
for (int j = ; j < n; j++) {
ans += j - sum(a[j]);
add(a[j], );
}
printf("%lld\n", ans);
}
树状数组可以高效地求出连续的一段元素之和或者更新单个元素的值。但是无法高效地给某一个区间里的所有元素同时加上一个值。因此,本题无法直接使用树状数组。在这里先考虑利用线段树来求解,然后再考虑改造树状数组来求解。
对于每个节点,维护有该节点对应的区间的和,那么就可以在 O(log n)时间内求出任意区间的和。但这样就没有办法高效地实现对一个区间同时加一个值,因为需要对这个区间相关的所有节点都进行更新才可以。
为了保持线段树的高效,对于每个节点我们维护以下两个数据
(1)给这个节点对应的区间内的所有元素共同加上的值
(2)在这个节点对应的区间中除去(1)之外其他的值的和
通过单独维护共同加上的值,给区间同时加上一个值的操作就可以高效地进行了。如果对于父亲节点同时加了一个值,那么这个值就不会在儿子节点被重复考虑。在递归计算和时再把这一部分的值加到结果里面就可以了。这样,不论是同时加一个值还是查询一段的和的复杂度都是O(log n)。
(1)T 是操作的种类。第 i 个操作的 T[ i ] 是 C 的话,就是给区间同时加上一个值,是 Q 的话则是查询一段的和。
(2)A, L, R 都是以 0 为下标起点的。
typedef long long ll;
const int DAT_SIZE = ( << ) - ; int N, Q;
int A[MAX_N];
char T[MAX_Q];
int L[MAX_Q], R[MAX_Q], X[MAX_Q]; // 线段树
ll data[DATA_SIZE], datb[MAX_SIZE]; // 对区间[a, b]同时加 x
// k 是节点的编号,对应的区间是(l, r)
void add(int a, int b, int x, int k, int l, int r) {
if (a <= l && r <= b)
data[k] += x;
else if (l < b && a < r) {
datb[k] += (min(b, r) - max(a, l)) * x;
add(a, b, x, k * + , l, (l + r) / );
add(a, b, x, k * + , (l + r) / , r);
}
} // 对区间[a, b]同时加 x
// k 是节点的编号,对应的区间是(l, r)
ll sum(int a, int b, int k, int l, int r) {
if (b <= || r <= a) return ;
else if (a <= && r <= b) return data[k] * (r - ) + data[k];
else {
ll res = (min(b, r) - max(a, l)) * data[k];
res += sum(a, b, k * + , l, (l + r) / );
res += sum(a, b, k * + , (l + r) / , r);
return res;
}
} void solve() {
for (int i = ; i < N; i++)
add(i, i + , A[i], , , N);
for (int i = ; i < Q; i++) {
if (T[i] == 'C')
add(L[i], R[i] + , X[i], , , N);
else
printf("%lld\n", sum(L[i], R[i] + , , , N));
}
}
未完待续
树状数组(Binary Indexed Tree)的更多相关文章
- 树状数组(Binary Indexed Tree) 总结
1.“树状数组”数据结构的一种应用 对含有n个元素的数组(a[1],...,a[k],...,a[n]): (1)求出第i个到第j个元素的和,sum=a[i]+...+a[j]. 进行j-i+1次加法 ...
- 树状数组 Binary Indexed Tree/Fenwick Tree
2018-03-25 17:29:29 树状数组是一个比较小众的数据结构,主要应用领域是快速的对mutable array进行区间求和. 对于一般的一维情况下的区间和问题,一般有以下两种解法: 1)D ...
- 树状数组(Binary Indexed Tree(BIT))
先不说别的,这个博客为我学习树状数组提供了很大帮助,奉上传送门 http://blog.csdn.net/int64ago/article/details/7429868 然后就说几个常用的操作 in ...
- 树状数组(Binary Index Tree)
一维BIT(单点更新,区间求和): Problem - 1166 #include <iostream> #include <algorithm> #include <c ...
- 树状数组,Fenwick Tree
Fenwick Tree, (also known as Binary Indexed Tree,二叉索引树), is a high-performance data structure to cal ...
- 树形DP+DFS序+树状数组 HDOJ 5293 Tree chain problem(树链问题)
题目链接 题意: 有n个点的一棵树.其中树上有m条已知的链,每条链有一个权值.从中选出任意个不相交的链使得链的权值和最大. 思路: 树形DP.设dp[i]表示i的子树下的最优权值和,sum[i]表示不 ...
- HDU 3436--Queue-jumpers (树状数组 or Splay Tree)
树状数组这个真心想了好久,还是没想出来 %%% www.cppblog.com/Yuan/archive/2010/08/18/123871.html 树状数组求前缀和大于等于k的最大值,第一次看到这 ...
- 树状数组(fenwick tree)
树状数组又称芬威克树,概念上是树状,实际上是使用数组实现的,表现为一种隐式数据结构,balabala...详情请见:https://en.wikipedia.org/wiki/Fenwick_tree ...
- NYOJ 108 士兵杀敌1(树状数组)
首先,要先讲讲树状数组: 树状数组(Binary Indexed Tree(BIT), Fenwick Tree)是一个查询和修改复杂度都为log(n)的数据结构.主要用于查询任意两位之间的所有元素之 ...
- 树状数组-HDU1541-Stars一维树状数组 POJ1195-Mobile phones-二维树状数组
树状数组,学长很早之前讲过,最近才重视起来,enmmmm... 树状数组(Binary Indexed Tree(B.I.T), Fenwick Tree)是一个查询和修改复杂度都为log(n)的数据 ...
随机推荐
- Dialupass v3.20 汉化绿色版 显示查看拨号上网密码
Dialupass 显示查看拨号上网密码 拨号上网的密码不小心丢了怎么办?这个工具可以帮你!在紧要关头,它会让你体验到它的奇效!有备无患,快收藏这个小东东吧. 这是一款拯救忘记了拨号网络密码的使用者的 ...
- linux netstat 查看端口
1. netstat命令用于显示系统的网络信息,包括网络连接 .路由表 .接口状态2. 一般我们使用 netstat 来查看本机开启了哪些端口,查看有哪些客户端连接 [root@localhost ~ ...
- mac os cmake安装
1.下载安装程序,地址为 http://www.cmake.org/download/,下载 Unix/Linux Source (has \n line feeds) cmake-3.0.2.tar ...
- Windows下备份mysql
---恢复内容开始--- Windows下备份mysql 第一步 编写脚本 ::设置时间变量 set "Ymd=%date:~0,4%%date:~5,2%%date:~8,2%%time ...
- set的常见用法
set的使用 set是什么 set是一个内部有序且不含重复元素的容器 用处 *使得元素自动有序 *去除重复元素 set的引入 # include <set> using namespace ...
- SPOJ 4003 Phone List 题解
题面 啊~,很水的一道trie树模板题: 当两个串存在关系时情况有两种: 若当前串插入后没有任何新建节点,则该串肯定是之前插入的某个串的前缀: 若在插入的时候,有某个经过的节点带有某串结尾的标记,则之 ...
- 杜恩德的新博客,都来看看-duende99
啊啊啊啊 https://home.cnblogs.com/u/duende99/
- neo4j开发自定义存储过程注意事项
开发自定义的neo4j存储过程(procedures)注意事项及说明: 1.调用方式: 在neo4j的web界面(http://localhost:7474/)命令行输入框内,输入call your_ ...
- RabbitMQ入门教程(九):首部交换机Headers
原文:RabbitMQ入门教程(九):首部交换机Headers 版权声明:本文为博主原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明. 本文链接:https://blog ...
- 3.学习Dispatcher
3.学习Dispatcher 不管是WinForm应用程序还是WPF应用程序,实际上都是一个进程,一个进程可以包含多个线程,其中有一个是主线程,其余的是子线程. 在WPF或WinForm应用程序中,主 ...