树状数组 Binary Indexed Tree/Fenwick Tree
2018-03-25 17:29:29
树状数组是一个比较小众的数据结构,主要应用领域是快速的对mutable array进行区间求和。
对于一般的一维情况下的区间和问题,一般有以下两种解法:
1)DP
预处理:建立长度为n的数组,每个结点i保存前i个数的和,时间复杂度O(n)。
查询:直接从数组中取两个段相减,时间复杂度O(1)。
更新:这种方法比较适用与immutable数组,对于mutable数组的更新需要重新建立表,所以时间复杂度为O(n)。
2)树状数组 BIT
预处理:建立树状数组,对数组中的每个数进行update操作,时间复杂度O(nlogn)。
查询:从当前结点向根结点遍历,求总和,时间复杂度O(logn)。
更新:从当前结点向根结点遍历,更新这条路径上的所有结点的值,时间复杂度O(logn)。
一、一维树状数组
在DP算法中我们在每个结点存放的是当前结点与前面所有结点的和,这就直接导致了在做更新的时候我们也只能进行大规模的修正,树状数组的提出就是为了在更新的过程中也保证有较低的时间复杂度,要实现这个目的,显然,每个结点我们不能再存储全部的前i个数,只能进行部分存储,而这部分存储的个数,就是整个树状数组的核心和关键。
在树状数组中,每个结点存的和的个数是lowbit(i)个,这里的lowbit就是i的二进制表示的第一个1所表示的数,举例:4(0100),lowbit(4) = 100,也就是在4号结点位置要存储4个结点的和。而这3个数(自己除外)都是4号结点的子孙,这三个数显然是0011,0010,0001。
那么,parent结点和child结点到底有什么关系呢?在树状数组中,我们规定parent = child + lowbit(child)。在更新操作中,我们可以递归的向上遍历,将所有该结点的父亲结点都进行更新,时间复杂度为O(logn)。
那么,又如何进行查询呢?对于sum(1, j) = nums[1] + nums[2] + ... + nums[j]。由于在树状数组中j号结点中存了lowbit(j)个数的和,所以原式可以写成sum(1, j) = sum(1, j - lowbit(j)) + BIT(j)。因此也可以进行递归或者迭代的求解。更进一步的分析,我们可以得知在查询的过程中,其实也生成了一组树,如下图。

public class BinaryIndexedTree1D {
int[] BIT;
BinaryIndexedTree1D(int n) {
this.BIT = new int[n + 1];
}
void update(int index, int delta) {
for (int i = index; i < BIT.length; i += (i & -i)) {
BIT[i] += delta;
}
}
int query(int index) {
int res = 0;
for (int i = index; i > 0; i -= (i & -i)) {
res += BIT[i];
}
return res;
}
}
问题描述:

问题求解:
public class NumArray {
FenwickTree ft;
int[] ls;
public NumArray(int[] nums) {
ft = new FenwickTree(nums.length);
ls = nums;
for (int i = 0; i < ls.length; i++) {
ft.update(i + 1, ls[i]);
}
}
public void update(int i, int val) {
ft.update(i + 1, val - ls[i]);
ls[i] = val;
}
public int sumRange(int i, int j) {
return ft.query(j + 1) - ft.query(i);
}
}
class FenwickTree {
int[] BIT;
FenwickTree(int n) {
this.BIT = new int[n + 1];
}
void update(int index, int delta) {
for (int i = index; i < BIT.length; i += (i & -i)) {
BIT[i] += delta;
}
}
int query(int index) {
int res = 0;
for (int i = index; i > 0; i -= (i & -i)) {
res += BIT[i];
}
return res;
}
}
2019.04.28
class NumArray {
int[] bit;
int[] numsCopy;
int n;
public NumArray(int[] nums) {
bit = new int[nums.length + 1];
numsCopy = new int[nums.length];
n = nums.length + 1;
for (int i = 0; i < nums.length; i++) {
update(i, nums[i]);
numsCopy[i] = nums[i];
}
}
public void update(int i, int val) {
int idx = i + 1;
int delta = val - numsCopy[i];
numsCopy[i] = val;
for (int k = ++i; k < n; k += (k & -k)) {
bit[k] += delta;
}
}
private int query(int i) {
int res = 0;
for (int k = i; k > 0; k -= (k & -k)) {
res += bit[k];
}
return res;
}
public int sumRange(int i, int j) {
return query(j + 1) - query(i);
}
}
二、二维树状数组
二维的树状数组其实就是分别对每行每列进行树状数组化,编写代码上面和一维数组是非常类似的。
public class BinaryIndexedTree2D {
int[][] BIT;
BinaryIndexedTree2D(int N, int M) {
BIT = new int[N + 1][M + 1];
}
void update(int n, int m, int delta) {
for (int i = n; i < BIT.length; i += (i & -i)) {
for (int j = m; j < BIT[0].length; j += (j & -j)) {
BIT[i][j] += delta;
}
}
}
int query(int n, int m) {
int res = 0;
for (int i = n; i > 0; i -= (i & -i)) {
for (int j = m; j > 0; j -= (j & -j)) {
res += BIT[i][j];
}
}
return res;
}
}
问题描述:

问题求解:
裸的2维树状数组问题,有个坑是所有的数字都需要对1e9进行取模,并且最终的sum结果不能为负数。
import java.util.Scanner;
public class Main {
public static void main(String[] args) {
int mod = (int)1e9 + 7;
Scanner sc = new Scanner(System.in);
String s = sc.nextLine();
int N = Integer.valueOf(s.split(" ")[0]);
int M = Integer.valueOf(s.split(" ")[1]);
BinaryIndexedTree2D bit = new BinaryIndexedTree2D(N, N);
for (int k = 0; k < M; k++) {
s = sc.nextLine();
String[] ls = s.split(" ");
if (ls[0].equals("Add")) {
int i = Integer.valueOf(ls[1]);
int j = Integer.valueOf(ls[2]);
int val = Integer.valueOf(ls[3]);
bit.update(i + 1, j + 1, val);
}
else if (ls[0].equals("Sum")) {
int x1 = Integer.valueOf(ls[1]);
int y1 = Integer.valueOf(ls[2]);
int x2 = Integer.valueOf(ls[3]);
int y2 = Integer.valueOf(ls[4]);
System.out.println((bit.query(x2 + 1, y2 + 1) - bit.query(x2 + 1, y1) - bit.query(x1, y2 + 1) + bit.query(x1, y1) + mod) % mod);
}
}
}
}
class BinaryIndexedTree2D {
int[][] BIT;
int mod;
BinaryIndexedTree2D(int N, int M) {
BIT = new int[N + 1][M + 1];
mod = (int)1e9 + 7;
}
void update(int n, int m, int delta) {
for (int i = n; i < BIT.length; i += (i & -i)) {
for (int j = m; j < BIT[0].length; j += (j & -j)) {
BIT[i][j] = (BIT[i][j] + delta) % mod;
}
}
}
int query(int n, int m) {
int res = 0;
for (int i = n; i > 0; i -= (i & -i)) {
for (int j = m; j > 0; j -= (j & -j)) {
res = (res + BIT[i][j]) % mod;
}
}
return res;
}
}
三、逆序对问题
逆序对问题是一个经典的问题,使用树状数组可以很好的解决这个问题。
树状数组的核心是单点更新,区间求和。
问题的核心就变成了如何将逆序对问题转化程区间求和的问题,简单的转化方式有,构建一个频率计数桶,将出现的元素放到相应的桶中,并将桶中的数量加一。从后向前逆序遍历数组,边遍历边更新桶中的数量,当遍历到一个元素的时候,计算getSum(num - 1)就可以得到当前元素的逆序对个数。
这个方法的问题就是单纯采用数字的大小来建立桶的话,这个桶的范围可能会很大,其实我们需要的只是相对的大小,所以我们可以将nums mapping 到sort后的idx上,这样整个的空间复杂度就降到了unique num的数量级。
问题描述:

问题求解:
public List<Integer> countSmaller(int[] nums) {
List<Integer> res = new ArrayList<>();
int[] sorted = Arrays.copyOf(nums, nums.length);
Arrays.sort(sorted);
Map<Integer, Integer> ranks = new HashMap<>();
int rank = 0;
for (int i = 0; i < sorted.length; ++i)
if (i == 0 || sorted[i] != sorted[i - 1])
ranks.put(sorted[i], ++rank);
int[] bit = new int[ranks.size() + 1];
for (int i = nums.length - 1; i >= 0; --i) {
int sum = query(bit, ranks.get(nums[i]) - 1);
res.add(sum);
update(bit, ranks.get(nums[i]), 1);
}
Collections.reverse(res);
return res;
}
private int query(int[] bit, int i) {
int res = 0;
for (int k = i; k > 0; k -= (k & -k)) {
res += bit[k];
}
return res;
}
private void update(int[] bit, int i, int delta) {
for (int k = i; k < bit.length; k += (k & -k)) {
bit[k] += delta;
}
}
树状数组 Binary Indexed Tree/Fenwick 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(BIT))
先不说别的,这个博客为我学习树状数组提供了很大帮助,奉上传送门 http://blog.csdn.net/int64ago/article/details/7429868 然后就说几个常用的操作 in ...
- 树状数组(Binary Index Tree)
一维BIT(单点更新,区间求和): Problem - 1166 #include <iostream> #include <algorithm> #include <c ...
- 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)的数据 ...
- 树状数组入门 hdu1541 Stars
树状数组 树状数组(Binary Indexed Tree(B.I.T), Fenwick Tree)是一个查询和修改复杂度都为log(n)的数据结构.主要用于查询任意两位之间的所有元素之和,但是每次 ...
- 14-敌兵布阵(HDU1166线段树 & 树状数组)
http://acm.hdu.edu.cn/showproblem.php?pid=1166 敌兵布阵 Time Limit: 2000/1000 MS (Java/Others) Memory ...
- HDU 1166 敌兵布阵 树状数组小结(更新)
树状数组(Binary Indexed Tree(BIT), Fenwick Tree) 是一个查询和修改复杂度都为log(n)的数据结构.主要用于查询任意两位之间的所有 元素之和,但是每次只能修改一 ...
- HDU 1541.Stars-一维树状数组(详解)
树状数组,学长很早之前讲过,最近才重视起来,enmmmm... 树状数组(Binary Indexed Tree(B.I.T), Fenwick Tree)是一个查询和修改复杂度都为log(n)的数据 ...
随机推荐
- Java项目工程化之项目构建工具Maven
欢迎查看Java开发之上帝之眼系列教程,如果您正在为Java后端庞大的体系所困扰,如果您正在为各种繁出不穷的技术和各种框架所迷茫,那么本系列文章将带您窥探Java庞大的体系.本系列教程希望您能站在上帝 ...
- Oracle命令(三):Oracle用户
1.显示当前用户名 select user from dual; show user 2.显示当然用户有哪些表 select * from tab; 3.显示当所有用户的表 select * from ...
- NSString 属性为啥用copy 不用strong
copy不能修改,strong可以修改,防止字符串被意外修改.demo: ——————————————————code 你要的 demo—————————————————— @property (n ...
- Python开发【笔记】:git&github 快速入门
github入门 简介: 很多人都知道,Linus在1991年创建了开源的Linux,从此,Linux系统不断发展,已经成为最大的服务器系统软件了. Linus虽然创建了Linux,但Linux的壮大 ...
- mysql 数据操作 多表查询 子查询 介绍
子查询就是: 把一条sql语句放在一个括号里,当做另外一条sql语句查询条件使用 拿到这个结果以后 当做下一个sql语句查询条件mysql 数据操作 子查询 #1:子查询是将一个查询语句嵌套在另一个 ...
- Android仿今日头条手界面
public class MyIndicator extends HorizontalScrollView implements ViewPager.OnPageChangeListener { pr ...
- centos shell编程5 LANMP一键安装脚本 lamp sed lnmp 变量和字符串比较不能用-eq cat > /usr/local/apache2/htdocs/index.php <<EOF重定向 shell的变量和函数命名不能有横杠 平台可以用arch命令,获取是i686还是x86_64 curl 下载 第三十九节课
centos shell编程5 LANMP一键安装脚本 lamp sed lnmp 变量和字符串比较不能用-eq cat > /usr/local/apache2/htdocs/ind ...
- ssh 配置文件讲解大全 ssh调试模式 sftp scp strace进行调试 特权分离
ssh 配置文件讲解大全 ssh调试模式 sftp scp strace进行调试 特权分离 http://blog.chinaunix.net/uid-16728139-id-3265394.h ...
- git-【四】撤销修改和删除文件操作
一:撤销修改: 比如我现在在readme.txt文件里面增加一行 内容为555555555555,我们先通过命令查看如下: 在未提交之前,发现添加5555555555555内容有误,所以得马上恢复以前 ...
- t检验&z检验学习[转载]
转自:https://blog.csdn.net/m0_37777649/article/details/74937242 1.什么是T检验? T检验是假设检验的一种,又叫student t检验(St ...