通俗易懂的树状数组详解!请食用!qwq

2023-11-01 23:08

本文主要是介绍通俗易懂的树状数组详解!请食用!qwq,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前:我写的线段树深受好评。于是想写树状数组。持续更新。

树状数组与二进制位有关。和线段树有些相像之处,都是把序列转化成一棵树进行操作。但是有区别。

update 2021.1.3:发现咕了接近2个月,决定写写

part 1 :基本思想

1 二进制分解

如果1个数 a a a 的二进制表示中,为1的为分别为:
a k , 1 , a k , 2 , a k , 3 . . . a k , c n t a_{k,1},a_{k,2},a_{k,3}...a_{k,cnt} ak,1,ak,2,ak,3...ak,cnt

则这个数=
2 k , 1 + 2 k , 2 . . . + 2 k , c n t 2^{k,1}+2^{k,2}...+2^{k,cnt} 2k,1+2k,2...+2k,cnt

这就是二进制分解了。

2 树的由来

我们假设 k 1 > k 2 > k 3 . . . > k c n t k_1>k_2>k_3...>k_{cnt} k1>k2>k3...>kcnt,则区间 [ 1 , n ] [1,n] [1,n] 可以被分成 O ( l o g n ) O(log n) O(logn) 个小区间。

它们分别为:
[ 1 , 2 k 1 ] , [ 2 k 1 + 1 , 2 k 1 + 2 k 2 ] . . . [1,2^{k_1}],[2^{k_1}+1,2^{k_1}+2^{k_2}]... [1,2k1],[2k1+1,2k1+2k2]...

于是有了一棵树。

3 小区间特点

如果区间结尾为 r r r,则区间长度就等于 r r r 的二进制分解中最小的2的次幂。我们用函数 l o w b i t lowbit lowbit来求区间长度。

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

有了这个函数,我们就可以用类似标记的整块修改维护前缀和啦!!!

4 树的性质

我们用数组 s u m [ x ] sum[x] sum[x] 来保存序列 a a a 的区间 [ x − l o w b i t ( x ) + 1 , x ] [x-lowbit(x)+1,x] [xlowbit(x)+1,x] 中所有数的和,则数组 s u m sum sum 就是一个树形结构。

这就是我们的树状数组啦!

图解:

图中红色节点忽略不计qwq

在这个结构中:

1.每个节点保存以它为根的子树中所有叶节点的和

2.每个节点 s u m [ x ] sum[x] sum[x] 的子节点(指最底层的叶子结点!)个数等于 l o w b i t ( x ) lowbit(x) lowbit(x) 的值。

笔者的书本上没有括号内容,然而我经过研究后发现,是指叶子结点的值。如4有1.2.3.4这4个叶子结点,对吧?所以 l o w b i t ( 4 ) = 4 lowbit(4)=4 lowbit(4)=4 。但是4有7个子节点(A4个,C3个)。

3.除了根节点,每个节点 s u m [ x ] sum[x] sum[x] 的父亲节点是 s u m [ x + l o w b i t ( x ) ] sum[x+lowbit(x)] sum[x+lowbit(x)]

4.一整颗树的深度为 O ( l o g N ) O(log N) O(logN)

基本思想讲完了!

part 2 :常支持的操作

树状数组资瓷查询前缀和(自然可以区间和啦!),单点修改,求逆序对(和正序对),区间修改等。

好的,下一部分qwq。

part 3 :代码

像这种数据结构(线段树,分块…)基本都有整体修改的标记,对吧qwq?

前缀和:

遍历一个节点的子树中的子节点,就可以了。

int ask(int x){int res=0;for(;x;x-=lowbit(x))res+=sum[x];return res;
} 

单点修改:

这个操作除了修改,还要维护前缀和,所以我们要对一个节点的祖先进行遍历,从下往上修改。这样子修改完,前缀和一定也是正确维护的。

void add(int x,int val){for(;x<=n;x+=lowbit(x))sum[x]+=val;
}//n为原始数列中元素个数 

区间修改:

我们新建一个数组 b b b,把一次区间修改(在区间 [ l , r ] [l,r] [l,r] 中,把每一个值加上 v a l val val)转化为:

b [ l ] b[l] b[l] 处加上 v a l val val,在 b [ r + 1 ] b[r+1] b[r+1] 处减去 v a l val val

原理是前缀和。在区间 [ 1 , l − 1 ] [1,l-1] [1,l1] 中,前缀和不变;在区间 [ l , r ] [l,r] [l,r] 中,前缀和增加了 v a l val val;在区间 [ r + 1 , n ] [r+1,n] [r+1,n] 中,前缀和不变(加 v a l val val 和减去 v a l val val 抵消)。

正确性使然。

这里的加是指单点修改。

不要直接 s u m [ x ] + = v a l sum[x]+=val sum[x]+=val,会气死我的

区间和(区间查询):

这种题的区间修改和上面类似,但是要改一下,因为还要执行区间和操作。我们使用两个树状数组 c 1 c1 c1 c 2 c2 c2

它们初始值为0。

对于修改区间 [ l , r ] [l,r] [l,r],我们执行四次操作:

c 1 c1 c1 中,add(l,val);add(r+1,-val);

c 2 c2 c2 中,add(l,l*val);add(r+1,-(r+1)*val);

我们还要用 一个 数组储存数列原本的前缀和。

下面,对于每一个区间查询,我们分成1到 r r r 和1到 l − 1 l-1 l1两个部分,查询结果就是二者相减。

int ask_lr(int l,int r){int ans=sum[r]+(r+1)*ask_c1(r)-ask_c2(r);ans-=(sum[l-1]+l*ask_c1(l-1)-ask_c2(l-1));return ans;
} //ask_c1是对c1的操作,ask_c2是对c2的操作,sum记录原始前缀和

这个式子好好理解下吧!

讲完啦!(注意:根据不同的题目,大部分时候请开long long)

part 4 :练习题

一个比较不错的题单,做做吧

结:

写了怎么久,终于写完啦!

谢谢阅读。

这篇关于通俗易懂的树状数组详解!请食用!qwq的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/326398

相关文章

MySQL 中的 CAST 函数详解及常见用法

《MySQL中的CAST函数详解及常见用法》CAST函数是MySQL中用于数据类型转换的重要函数,它允许你将一个值从一种数据类型转换为另一种数据类型,本文给大家介绍MySQL中的CAST... 目录mysql 中的 CAST 函数详解一、基本语法二、支持的数据类型三、常见用法示例1. 字符串转数字2. 数字

SpringBoot中SM2公钥加密、私钥解密的实现示例详解

《SpringBoot中SM2公钥加密、私钥解密的实现示例详解》本文介绍了如何在SpringBoot项目中实现SM2公钥加密和私钥解密的功能,通过使用Hutool库和BouncyCastle依赖,简化... 目录一、前言1、加密信息(示例)2、加密结果(示例)二、实现代码1、yml文件配置2、创建SM2工具

MyBatis-Plus 中 nested() 与 and() 方法详解(最佳实践场景)

《MyBatis-Plus中nested()与and()方法详解(最佳实践场景)》在MyBatis-Plus的条件构造器中,nested()和and()都是用于构建复杂查询条件的关键方法,但... 目录MyBATis-Plus 中nested()与and()方法详解一、核心区别对比二、方法详解1.and()

Spring IoC 容器的使用详解(最新整理)

《SpringIoC容器的使用详解(最新整理)》文章介绍了Spring框架中的应用分层思想与IoC容器原理,通过分层解耦业务逻辑、数据访问等模块,IoC容器利用@Component注解管理Bean... 目录1. 应用分层2. IoC 的介绍3. IoC 容器的使用3.1. bean 的存储3.2. 方法注

MySQL 删除数据详解(最新整理)

《MySQL删除数据详解(最新整理)》:本文主要介绍MySQL删除数据的相关知识,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录一、前言二、mysql 中的三种删除方式1.DELETE语句✅ 基本语法: 示例:2.TRUNCATE语句✅ 基本语

Python内置函数之classmethod函数使用详解

《Python内置函数之classmethod函数使用详解》:本文主要介绍Python内置函数之classmethod函数使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录1. 类方法定义与基本语法2. 类方法 vs 实例方法 vs 静态方法3. 核心特性与用法(1编程客

Python函数作用域示例详解

《Python函数作用域示例详解》本文介绍了Python中的LEGB作用域规则,详细解析了变量查找的四个层级,通过具体代码示例,展示了各层级的变量访问规则和特性,对python函数作用域相关知识感兴趣... 目录一、LEGB 规则二、作用域实例2.1 局部作用域(Local)2.2 闭包作用域(Enclos

Python实现对阿里云OSS对象存储的操作详解

《Python实现对阿里云OSS对象存储的操作详解》这篇文章主要为大家详细介绍了Python实现对阿里云OSS对象存储的操作相关知识,包括连接,上传,下载,列举等功能,感兴趣的小伙伴可以了解下... 目录一、直接使用代码二、详细使用1. 环境准备2. 初始化配置3. bucket配置创建4. 文件上传到os

MySQL查询JSON数组字段包含特定字符串的方法

《MySQL查询JSON数组字段包含特定字符串的方法》在MySQL数据库中,当某个字段存储的是JSON数组,需要查询数组中包含特定字符串的记录时传统的LIKE语句无法直接使用,下面小编就为大家介绍两种... 目录问题背景解决方案对比1. 精确匹配方案(推荐)2. 模糊匹配方案参数化查询示例使用场景建议性能优

Java内存分配与JVM参数详解(推荐)

《Java内存分配与JVM参数详解(推荐)》本文详解JVM内存结构与参数调整,涵盖堆分代、元空间、GC选择及优化策略,帮助开发者提升性能、避免内存泄漏,本文给大家介绍Java内存分配与JVM参数详解,... 目录引言JVM内存结构JVM参数概述堆内存分配年轻代与老年代调整堆内存大小调整年轻代与老年代比例元空