1、树状数组
树状数组专题:https://leetcode-cn.com/tag/binary-indexed-tree/problemset/
视频:https://www.bilibili.com/video/BV1LW411e7jo?spm_id_from=333.337.search-card.all.click
2、题目描述
https://leetcode-cn.com/problems/range-sum-query-mutable/
3、代码详解
https://leetcode-cn.com/problems/range-sum-query-mutable/solution/qu-yu-he-jian-suo-shu-zu-ke-xiu-gai-by-l-76xj/
class NumArray {
private int[] nums; // 题目要求实现更新 nums 在某个位置的值,因此保存原始的 nums 数组
private int[] tree;
private int lowBit(int x) {
return x & -x;
}
// 单点修改: 在树状数组 index 位置中增加值 val
private void add(int index, int val) {
while (index < tree.length) {
tree[index] += val;
index += lowBit(index);
}
}
// 区间查询:查询前 index 个元素的前缀和
private int prefixSum(int index) {
int sum = 0;
while (index > 0) {
sum += tree[index];
index -= lowBit(index);
}
return sum;
}
//————————————————————————————————
// 构造函数
public NumArray(int[] nums) {
this.tree = new int[nums.length + 1];
this.nums = nums;
for (int i = 0; i < nums.length; i++) {
// 树状数组是一种可以动态维护序列前缀和的数据结构(序列下标从 1 开始)
add(i + 1, nums[i]);
}
}
// 更新 数组 nums 下标对应的值: 将 nums[index] 的值 更新 为 val
public void update(int index, int val) {
add(index + 1, val - nums[index]);
nums[index] = val;
}
// 返回数组 nums 中索引 left 和索引 right 之间( 包含 )的nums元素的 和
public int sumRange(int left, int right) {
return prefixSum(right + 1) - prefixSum(left);
}
}
/**
* Your NumArray object will be instantiated and called as such:
* NumArray obj = new NumArray(nums);
* obj.update(index,val);
* int param_2 = obj.sumRange(left,right);
*/