IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> 线段树详解 -> 正文阅读

[数据结构与算法]线段树详解

线段树是一种很重要的数据结构,在ACM中用的较多。用于对数据的更新和查询,主要优势体现在对段的处理上。比如:要将数组a从a【i】~a【j】的元素均加上一个数b,那么将做(j-i+1)次。如果有一个段就表示a【i】~a【j】,那么直接将这个段上的和值sum加上b*(j-i+1)即可,仅需操作一次。(这里只关心一个段的和)。再者,若想要知道a【i】~a【j】的和,需查询a【i】,a【i+1】...a【j】,再将其相加,也就是做(j-i+1)次,如果查询段a【i】~a【j】,直接取出该段的sum即可,当(j-i+1)很大时,查询这个段的复杂度也会远远低于前面的做法。那么,怎样来表示这个一段呢?要表示多少个段呢?下面介绍线段树是怎么做的。

节点定义如下:

struct node{
    int l;       //线段的左端点
    int r;       //线段的右端点
    int value;   //这个线段中各元素的和
};

对于每个节点,应该还有两个指针来表示该节点的左右孩子,用一个数组tree【】来模拟一个树,那么,对于节点tree【v】,它的左孩子节点为tree【i*2】,右孩子节点为tree【2*i+1】。那么左右孩子是什么呢?在线段树中,左右孩子分别是父节点的子区间,如父节点代表【l,r】,

令 int mid=(l+r)/2;则左孩子表示的区间为【l,mid】,右孩子表示的区间为【mid+1,r】。叶子节点x表示区间是【x,x】的一个值,即是原数组中的一个元素。

那么对于value值的设置又是怎么样的呢?是从叶子节点递归设置这个value。直接看代码理解

//建树 节点为v,q区间为[l,r]
void build(int v,int l,int r){ 
	tree[v].l=l;
	tree[v].r=r;
	if(l==r){
		//节点初始化
		tree[v].value=a[r];
		return ;
	}
	int mid=(l+r)>>1;
	build(2*v,l,mid);
	build(2*v+1,mid+1,r);
	//根据左右儿子更新节点value值
	tree[v].value=tree[2*v].value+tree[2*v+1].value;
}

1.更新

更新即维护树上的value值。

在对某一段a【i】~a【j】上的所有元素都加上一个值c的时候,在线段树上是怎么样的呢?首先找到这个段(可能是几个分段,比如找到两个分段(i~k)(k+1~j)),将每个段上的值都加上(c*(r-l+1)),r和l表示这个段上的区间,(r-l+1)表示这个区间的长度。这样是不是就完成了呢?不!比如给2到5这个段上加上了4*c,那么,当下次查询2到3和的时候,这个加上c*2的操作并没有体现(该操作只在2到5及其祖辈上有体现),那么怎么解决这个问题呢?

有一种做法是找到2到5结点更新后,再对其所有子孙结点进行更新,即也会给表示2到3的结点加上c*2,但是这种做法不好。对长度为n的区间i~j进行更新,复杂度是O(nlogn),而直接在元素组更新只需要n,这样线段树在更新上就显得没有优势了,而这样在比赛中也往往会TLE。

下面介绍一种记录增量的方法,即给每个结点再加一个域int add;记录更新操作的增量c,初始时每个结点的add均为0,当对2~5区间进行更新操作后,给该结点的add加上一个值c。再下次要对2~3结点进行更新和查询时,再将add传下去。具体操作看代码:

void update(int v,int l,int r,int m){ //更新[l,r]加上数m
	if(tree[v].l==l&&tree[v].r==r){ //找到了并更新
		tree[v].value+=m*(l-r+1);
		tree[v].add=m;
		return ;
	}
	//将add传递给儿子
	if(tree[v].add){
		tree[2*v].add+=tree[v].add;
		tree[2*v+1].add+=tree[v].add;
		tree[v].add=0;//传递完了记得清0
	}
	int mid=(tree[v].l+tree[v].r)/2;
	if(r<=mid) update(2*v,l,r,m); //区间被左儿子覆盖
	else{
		if(l>mid){ //区间被右儿子覆盖
			update(v*2+1,l,r,m);
		}else{
			//横跨左右儿子
			update(v*2,l,mid,m);
			update(v*2+1,mid+1,r,m);
		}
	}
}

2.查询

即查询区间【l,r】上的value

//当前查询节点为v,区间为[l,r]
void query(int v,int l,int r){
	if(tree[v].l==l&&tree[v].r==r){
		ans+=tree[v].value;
		return ;
	}
	//看有没有没有传递完的add
	if(tree[v].add){
		tree[v*2].add+=tree[v].add;
		tree[v*2+1].add+=tree[v].add;
		tree[v].add=0;
	}
	int mid=(tree[v].l+tree[v].r)/2;
	if(r<=mid) query(v*2,l,r); //left
	else{
		if(l>mid) query(2*v+1,l,r); //right
		else{
			query(2*v,l,mid);
			query(2*v+1,mid+1,r);
		}
	}
}

线段树的用法非常灵活,离散化也是经常配合线段树使用的,因为线段树所需要的空间为4*MAX,,MAX为原数据的个数

  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2021-11-19 17:52:03  更:2021-11-19 17:53:41 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/9 1:58:12-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码