通俗易懂的树状数组详解!请食用!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

相关文章

Debezium 与 Apache Kafka 的集成方式步骤详解

《Debezium与ApacheKafka的集成方式步骤详解》本文详细介绍了如何将Debezium与ApacheKafka集成,包括集成概述、步骤、注意事项等,通过KafkaConnect,D... 目录一、集成概述二、集成步骤1. 准备 Kafka 环境2. 配置 Kafka Connect3. 安装 D

Java中ArrayList和LinkedList有什么区别举例详解

《Java中ArrayList和LinkedList有什么区别举例详解》:本文主要介绍Java中ArrayList和LinkedList区别的相关资料,包括数据结构特性、核心操作性能、内存与GC影... 目录一、底层数据结构二、核心操作性能对比三、内存与 GC 影响四、扩容机制五、线程安全与并发方案六、工程

C++初始化数组的几种常见方法(简单易懂)

《C++初始化数组的几种常见方法(简单易懂)》本文介绍了C++中数组的初始化方法,包括一维数组和二维数组的初始化,以及用new动态初始化数组,在C++11及以上版本中,还提供了使用std::array... 目录1、初始化一维数组1.1、使用列表初始化(推荐方式)1.2、初始化部分列表1.3、使用std::

C++ Primer 多维数组的使用

《C++Primer多维数组的使用》本文主要介绍了多维数组在C++语言中的定义、初始化、下标引用以及使用范围for语句处理多维数组的方法,具有一定的参考价值,感兴趣的可以了解一下... 目录多维数组多维数组的初始化多维数组的下标引用使用范围for语句处理多维数组指针和多维数组多维数组严格来说,C++语言没

Spring Cloud LoadBalancer 负载均衡详解

《SpringCloudLoadBalancer负载均衡详解》本文介绍了如何在SpringCloud中使用SpringCloudLoadBalancer实现客户端负载均衡,并详细讲解了轮询策略和... 目录1. 在 idea 上运行多个服务2. 问题引入3. 负载均衡4. Spring Cloud Load

Springboot中分析SQL性能的两种方式详解

《Springboot中分析SQL性能的两种方式详解》文章介绍了SQL性能分析的两种方式:MyBatis-Plus性能分析插件和p6spy框架,MyBatis-Plus插件配置简单,适用于开发和测试环... 目录SQL性能分析的两种方式:功能介绍实现方式:实现步骤:SQL性能分析的两种方式:功能介绍记录

在 Spring Boot 中使用 @Autowired和 @Bean注解的示例详解

《在SpringBoot中使用@Autowired和@Bean注解的示例详解》本文通过一个示例演示了如何在SpringBoot中使用@Autowired和@Bean注解进行依赖注入和Bean... 目录在 Spring Boot 中使用 @Autowired 和 @Bean 注解示例背景1. 定义 Stud

如何通过海康威视设备网络SDK进行Java二次开发摄像头车牌识别详解

《如何通过海康威视设备网络SDK进行Java二次开发摄像头车牌识别详解》:本文主要介绍如何通过海康威视设备网络SDK进行Java二次开发摄像头车牌识别的相关资料,描述了如何使用海康威视设备网络SD... 目录前言开发流程问题和解决方案dll库加载不到的问题老旧版本sdk不兼容的问题关键实现流程总结前言作为

SQL 中多表查询的常见连接方式详解

《SQL中多表查询的常见连接方式详解》本文介绍SQL中多表查询的常见连接方式,包括内连接(INNERJOIN)、左连接(LEFTJOIN)、右连接(RIGHTJOIN)、全外连接(FULLOUTER... 目录一、连接类型图表(ASCII 形式)二、前置代码(创建示例表)三、连接方式代码示例1. 内连接(I

Go路由注册方法详解

《Go路由注册方法详解》Go语言中,http.NewServeMux()和http.HandleFunc()是两种不同的路由注册方式,前者创建独立的ServeMux实例,适合模块化和分层路由,灵活性高... 目录Go路由注册方法1. 路由注册的方式2. 路由器的独立性3. 灵活性4. 启动服务器的方式5.