树状数组介绍

树状数组(Binary Indexed Tree,简称 BIT 或 Fenwick Tree)是一种可以高效解决前缀和问题的数据结构。它能在对数时间复杂度内完成单点更新和查询前缀和的操作。树状数组通过一种巧妙的方式,将数组元素的值分布在不同的位置上,使得查询和更新操作都能够在 O(log n) 的时间复杂度内完成。

树状数组的基本思想

树状数组将数组分为若干个“树状”的组,每个组内的元素更新会影响到多个父节点,查询前缀和时则只需要考虑这些父节点的值。具体来说,树状数组的每个节点 C[i] 维护的是原数组 A 中下标从 A[i-lowbit(i)+1] 到 A[i] 的所有元素的和。

其中,lowbit(i) 表示 i 的二进制表示中最低位的 1 及其后面的 0 所组成的数。例如,对于 i = 10(二进制 1010),lowbit(10) = 2(二进制 10)。

基本操作

1. 单点更新

要将原数组 A 中的某个元素 A[i] 更新为 v,需要更新树状数组中从 C[i] 开始,到所有包含 C[i] 的父节点为止的所有节点。

 

cpp复制代码

void update(int i, int delta) {
while (i < N) { // N 是数组的大小
C[i] += delta;
i += lowbit(i);
}
}
2. 查询前缀和

要查询原数组 A 中从 A[1] 到 A[i] 的所有元素的和,需要累加树状数组中从 C[i] 开始,到所有不包含 C[i] 的祖先节点为止的所有节点的值。

 

cpp复制代码

int query(int i) {
int sum = 0;
while (i > 0) {
sum += C[i];
i -= lowbit(i);
}
return sum;
}

lowbit 函数

lowbit 函数的作用是获取一个数的二进制表示中最低位的 1 及其后面的 0 所组成的数。可以通过位运算实现:

 

cpp复制代码

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

示例

假设有一个数组 A = [1, 2, 3, 4, 5],则对应的树状数组 C 可以通过以下方式构建:

  • C[1] = A[1] = 1
  • C[2] = A[1] + A[2] = 1 + 2 = 3
  • C[3] = A[3] = 3
  • C[4] = A[4] = 4
  • C[5] = A[5] = 5
  • C[6] = A[1] + A[2] + A[3] + A[4] + A[5] = 15
  • C[7] 及以上不需要(因为数组大小只有 5)

注意:在实际应用中,树状数组的索引通常从 1 开始,而不是从 0 开始,以避免处理边界情况。同时,为了方便计算,树状数组的大小通常设为比原数组大 1 的 2 的幂次方。

应用场景

树状数组常常用于解决需要频繁查询前缀和或进行单点更新的问题,如求部分和、区间和、动态数组查询等。由于它的高效性和易实现性,树状数组在编程竞赛和算法研究中有着广泛的应用。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/25932.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

无线领夹麦克风哪个品牌性价比高?推荐领夹麦克风性价比最高品牌

当今的直播、短视频已经深深的融入到了我们的生活当中&#xff0c;很多小伙伴会通过拍摄短视频、Vlog来分享自己生活精彩的瞬间。不过录制视频时&#xff0c;如果单纯靠手机拾音&#xff0c;距离手机越远&#xff0c;声音越小越不清晰如果有噪音干扰&#xff0c;那更是“惨不忍…

ONVIF相关介绍

ONVIF致力于通过全球性的开放接口标准来推进网络视频在安防市场的应用,这一接口标准将确保不同厂商生产的网络视频产品具有互通性。2008年11月,论坛正式发布了ONVIF第一版规范——ONVIF核心规范1.0。随着视频监控的网络化应用,产业链的分工将越来越细。有些厂商专门做摄像头…

spring和mybatis中的连接池和缓存

文章目录 十、连接池10.1连接池10.2、mybatis连接池的分类 十一、mybatis的缓存一级缓存和二级缓存使用一级缓存失效的四种情况&#xff1a;11.1、不同的SqlSession对应不同的一级缓存。11.2、MyBatis的二级缓存 二级缓存开启的条件&#xff1a;11、3二级缓存的相关配置11.4、m…

计算机视觉与深度学习实战,Python为工具,基于计算机视觉的自动驾驶应用

一、引言 随着人工智能技术的飞速发展,计算机视觉与深度学习已成为当今科技领域的热点。其中,自动驾驶技术作为二者的完美结合,已经引起了广泛关注。自动驾驶技术依赖于计算机视觉实现对周围环境的感知与理解,通过深度学习对大量数据进行处理和分析,从而实现无人驾驶的目标…

pytest中失败用例重跑

pip install pytest-rerunfailures 下载rerunfailures插件包 配置文件中加入命令 --reruns 次数 也可在命令行中pytest --rerun-failures2 可以在allure报告中看到重试效果

【Echarts系列】水平柱状图

【Echarts系列】水平柱状图 序示例数据格式代码 序 为了节省后续开发学习成本&#xff0c;这个系列将记录我工作所用到的一些echarts图表。 示例 水平柱状图如图所示&#xff1a; 数据格式 data [{name: 于洪区,value: 2736},{name: 新民市,value: 2844},{name: 皇姑区,…

使用v-for实现点击当前li,li背景颜色变为红色,其余颜色不变

在 Vue 3 中&#xff0c;可以使用 <script setup> 语法糖来简化组件的编写。 下面是一个使用 Vue 3 和 <script setup> 实现的示例&#xff1a; <template> <ul> <li v-for"(item, index) in items" :key"index" :cl…

C++:SLT容器-->queue

C:SLT容器-->queue 1. queue容器2. queue 常用接口 1. queue容器 先进先出队列允许从一端插入元素&#xff0c;从另一端删除元素队列中只有队头和队尾可以被外界使用&#xff0c;因此队列不允许有遍历行为队列中插入数据称为入队(push)&#xff0c;删除数据称为出队(pop) …

JS 实现动态规划

function getPaths(m, n) {// m * n 二维数组&#xff0c;模拟网格const map new Array(m)for (let i 0; i < m; i) {map[i] new Array(n)}// 如果只走第一行&#xff0c;就只有一条路径。所以第一行所有 item 都填充 1map[0].fill(1)// 如果只走第一列&#xff0c;也只有…

ADAS功能规范总览

前提 本文对ADAS算法相关功能规范进行详细分解&#xff0c;本专栏是对自动驾驶相关从业者进行入门指导&#xff0c;会对功能算法各个部分进行详细梳理&#xff0c;如有不同见解评论或私信交流。相关ADAS的简化版本规范请关注我的ADAS辅助驾驶算法专栏。 功能规范 会按照以下…

快速删除 node_modules

在Windows系统上删除 node_modules 文件夹通常会比较慢&#xff0c;主要原因有以下几个&#xff1a; 1.文件数量多且嵌套深&#xff1a;node_modules 文件夹通常包含大量的子文件夹和文件&#xff0c;其中一些可能嵌套得非常深。Windows在删除文件和文件夹时&#xff0c;需要遍…

【vue3|第8期】深入理解Vue 3 computed计算属性

日期&#xff1a;2024年6月10日 作者&#xff1a;Commas 签名&#xff1a;(ง •_•)ง 积跬步以致千里,积小流以成江海…… 注释&#xff1a;如果您觉得有所帮助&#xff0c;帮忙点个赞&#xff0c;也可以关注我&#xff0c;我们一起成长&#xff1b;如果有不对的地方&#xf…

合约ABI探秘

EOA使用ABI调用 合约之间使用接口调用 自己写ABI&#xff0c;自己写接口 ABI探秘 EOA使用的ABI数据与合约间调用时调用者持有的接口时等价的&#xff0c;都是对合约函数签名的完整描述调用者使用ABI数据生成message中的calldata message上下文的变化 合约之间的调用也模拟…

8.11 矢量图层线要素单一符号使用六(光栅线)

文章目录 前言光栅线&#xff08;Raster Line&#xff09;QGis设置线符号为光栅线&#xff08;Raster Line&#xff09;二次开发代码实现光栅线&#xff08;Raster Line&#xff09; 总结 前言 本章介绍矢量图层线要素单一符号中光栅线&#xff08;Raster Line&#xff09;的使…

node设置镜像源详细教程

在Node.js环境中&#xff0c;你可以通过设置npm或yarn的镜像源来加速依赖包的下载。以下是如何设置npm和yarn的镜像源的详细步骤&#xff1a; 使用npm设置镜像源 临时设置镜像源&#xff1a; 你可以在安装包时临时指定镜像源&#xff0c;例如&#xff1a; npm install package…

FastWeb网站开发之拦截器(interceptor)使用教程

拦截器用于拦截和处理HTTP请求和响应。在请求开始处理之前进行拦截过滤与处理。 API文档 拦截器一般用于初始化lua脚本中创建拦截规则。 它主要用于浏览器的请求中某些目录或文件不允许访问、某些目录需要验证权限&#xff0c;在拦截器中进行判断并选择是否继续向下执行。 …

在Spigot插件中配置指令

在Spigot插件中配置指令 在Spigot插件开发中&#xff0c;配置指令&#xff08;命令&#xff09;是非常重要的一部分。通过自定义指令&#xff0c;玩家可以与插件进行交互&#xff0c;触发特定的功能和行为。本文将详细介绍如何在Spigot插件中配置指令&#xff0c;并通过一个简…

【C语言】32个关键字

C语言32个关键字 文章目录 C语言32个关键字1.数据类型关键字基本数据类型&#xff08;5个&#xff09;类型修饰关键字&#xff08;4个&#xff09;复杂类型关键字&#xff08;5个&#xff09;存储级别关键字&#xff08;6个&#xff09; 2.流程控制关键字跳转结构&#xff08;4…

如何通过加密U盘 实现数据传输闭环管控?

加密U盘是用来保护存储在其中数据的安全的。通过加密技术&#xff0c;用户可以将其敏感文件和信息存储在U盘中&#xff0c;并设置密码或使用其他加密方法来防止未经授权的访问。这种安全措施可以防止数据泄露或盗窃&#xff0c;特别是在丢失或被盗的情况下&#xff0c;确保数据…

一些简单却精妙的算法

文章目录 1.树状数组2.红黑树3.星星打分4.欧几里得算法5.快速幂6.并查集 在编程的世界里&#xff0c;简洁的代码往往隐藏着深邃的智慧。一起来看看那些看似简单&#xff0c;实则精妙绝伦的代码片段&#xff0c;体会编程语言的优雅与力量。 1.树状数组 int lowbit(int x) { …