通俗易懂的树状数组详解!请食用!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数据库双机热备的配置方法详解

《MySQL数据库双机热备的配置方法详解》在企业级应用中,数据库的高可用性和数据的安全性是至关重要的,MySQL作为最流行的开源关系型数据库管理系统之一,提供了多种方式来实现高可用性,其中双机热备(M... 目录1. 环境准备1.1 安装mysql1.2 配置MySQL1.2.1 主服务器配置1.2.2 从

Linux kill正在执行的后台任务 kill进程组使用详解

《Linuxkill正在执行的后台任务kill进程组使用详解》文章介绍了两个脚本的功能和区别,以及执行这些脚本时遇到的进程管理问题,通过查看进程树、使用`kill`命令和`lsof`命令,分析了子... 目录零. 用到的命令一. 待执行的脚本二. 执行含子进程的脚本,并kill2.1 进程查看2.2 遇到的

MyBatis常用XML语法详解

《MyBatis常用XML语法详解》文章介绍了MyBatis常用XML语法,包括结果映射、查询语句、插入语句、更新语句、删除语句、动态SQL标签以及ehcache.xml文件的使用,感兴趣的朋友跟随小... 目录1、定义结果映射2、查询语句3、插入语句4、更新语句5、删除语句6、动态 SQL 标签7、ehc

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置

从基础到高级详解Go语言中错误处理的实践指南

《从基础到高级详解Go语言中错误处理的实践指南》Go语言采用了一种独特而明确的错误处理哲学,与其他主流编程语言形成鲜明对比,本文将为大家详细介绍Go语言中错误处理详细方法,希望对大家有所帮助... 目录1 Go 错误处理哲学与核心机制1.1 错误接口设计1.2 错误与异常的区别2 错误创建与检查2.1 基础

k8s按需创建PV和使用PVC详解

《k8s按需创建PV和使用PVC详解》Kubernetes中,PV和PVC用于管理持久存储,StorageClass实现动态PV分配,PVC声明存储需求并绑定PV,通过kubectl验证状态,注意回收... 目录1.按需创建 PV(使用 StorageClass)创建 StorageClass2.创建 PV

Python版本信息获取方法详解与实战

《Python版本信息获取方法详解与实战》在Python开发中,获取Python版本号是调试、兼容性检查和版本控制的重要基础操作,本文详细介绍了如何使用sys和platform模块获取Python的主... 目录1. python版本号获取基础2. 使用sys模块获取版本信息2.1 sys模块概述2.1.1

一文详解Python如何开发游戏

《一文详解Python如何开发游戏》Python是一种非常流行的编程语言,也可以用来开发游戏模组,:本文主要介绍Python如何开发游戏的相关资料,文中通过代码介绍的非常详细,需要的朋友可以参考下... 目录一、python简介二、Python 开发 2D 游戏的优劣势优势缺点三、Python 开发 3D

Redis 基本数据类型和使用详解

《Redis基本数据类型和使用详解》String是Redis最基本的数据类型,一个键对应一个值,它的功能十分强大,可以存储字符串、整数、浮点数等多种数据格式,本文给大家介绍Redis基本数据类型和... 目录一、Redis 入门介绍二、Redis 的五大基本数据类型2.1 String 类型2.2 Hash

Java中的.close()举例详解

《Java中的.close()举例详解》.close()方法只适用于通过window.open()打开的弹出窗口,对于浏览器的主窗口,如果没有得到用户允许是不能关闭的,:本文主要介绍Java中的.... 目录当你遇到以下三种情况时,一定要记得使用 .close():用法作用举例如何判断代码中的 input