Fenwick Tree

Fenwick Tree,又称树状数组,是一种二叉索引树,用于快速区间信息的维护和查询。它提供O(nlog n)的预处理和O(log n)的更新与查询操作。通过低bit技巧,可以实现add(x,d)更新和query(l,r)查询功能,解决了传统前缀和在更新后的效率问题。初始化时,可以通过add(i,A[i])完成,具有较高的时间复杂度效率。" 122744412,12120508,西门子SinuTrain最新版仿真软件操作与调试,"['西门子', '数控系统', 'FPGA开发', '软件仿真', 'CSS']

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Fenwick树,俗称树状数组

也就是二叉索引树(Binary Indexed Tree,BIT)

它的作用是什么呢

支持快速区间信息的维护和查询

为什么说是快速,因为它真的很快
O(nlog n)预处理
O(log n)维护和更新

先来看一个问题

给定一个n个元素的数组A
我们进行下面两个操作
1 add(x,d): 让A[x]增加d
2 query(l,r): 计算A[l]+A[l+1]+…+A[r]

怎么办

计算前缀和S,这可以在O(1)的时间解决query。但一旦进行add之后前缀后要重新算过。
不难发现如果进行add(x,d),对前面的前缀和是没有影响的,所以只要更新x之后的就好了
但其实还是不行,以为每次更新的花费还是很大
为什么花费会大
因为一个S的信息存储太大了,所以我们改小一点
但也不能太小,否则和直接算就一样了
所以我们用一个新的东西来存

下面来看一个图

注意到这里的C和我们之前的S很像
那我们就这样处理
用C来记一个和,但不是前缀和,而是类似与前缀和

仔细看这个图,如果我们要求前缀和,从C一直向右走,把走到的所有C加起来就是S

query解决了

再来看add

再看这个图,如果我们要更新一个点,从C一直向左走,把走到的所有C都加d就行了

那我们应该怎么走呢

先介绍一个东西lowbit
lowbi

Fenwick Tree,又称为树状数组(Binary Indexed Tree),是一种基于数组实现的数据结构,用于高效地动态维护前缀和。它可以在O(logn)的时间内完成以下操作:更新某个元素的值,查询某个区间的和。Fenwick Tree的实现原理是将数组分解为一系列的区间和,每个区间和保存在树状数组的相应位置上。通过使用二进制的技巧,可以高效地计算每个区间和。: ``` public class FenWickTree { private int[] values; private int[] bit; public FenWickTree(int length) { values = new int[length]; bit = new int[length + 1]; } public void setValues(int index, int value) { values[index = value; index += 1; while (index < bit.length) { bit[index += value; index += index & -index; } } public int getSum(int index) { int sum = 0; while (index > 0) { sum += bit[index]; index -= index & -index; } return sum; } } ``` 以上代码展示了如何使用Fenwick Tree实现动态维护前缀和的功能。其中setValues()方法用于更新某个元素的值,getSum()方法用于查询某个区间的和。 另外,Fenwick Tree也可以用来解决区间修改的问题。对于元素的修改,我们可以视为区间查询的逆过程,通过从叶节点开始向上更新父节点,依次对每个父节点进行相同的修改操作。具体的实现可以参考下面的示例代码: ``` class FenwickTree { private int[] tree; public FenwickTree(int n) { tree = new int[n + 1]; } public int lowbit(int x) { return x & (-x); } public void add(int i, int val) { while (i < tree.length) { tree[i += val; i += lowbit(i); } } public int query(int i) { int res = 0; while (i > 0) { res += tree[i]; i -= lowbit(i); } return res; } } ``` 这段代码展示了如何使用Fenwick Tree解决区间修改的问题。add()方法用于修改某个元素,query()方法用于查询某个区间的和。 综上所述,Fenwick Tree是一种用于高效地动态维护前缀和的数据结构,可以在O(logn)的时间内完成更新和查询操作。同时,它也可以应用于区间修改的问题。 #### 引用[.reference_title] - *1* *3* [【数据结构与算法】树状数组](https://blog.youkuaiyun.com/zzy_NIC/article/details/130616434)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *2* [6.10 Fenwick树](https://blog.youkuaiyun.com/m0_66201040/article/details/122923027)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值