线段树

线段树

线段树是一种二叉搜索树,与区间树相似,每一个叶子代表一个区间,它将一个区间划分成一些单元区间,每个单元区间对应线段树中的一个叶结点。


使用线段树可以快速的查找某一个节点在若干条线段中出现的次数,时间复杂度为O(logN)。而未优化的空间复杂度为2N,因此有时需要离散化让空间压缩。




如图所示,每一个节点都可以代表一个区间,子节点则分别表示父节点的左右半区间,例如父亲的区间是[a,b],那么(c=(a+b)/2)左儿子的区间是[a,c],右儿子的区间是[c+1,b]。下面看几种线段树的基本操作:

1.创建线段树

我们先定义一个用来储存数据的公用结构体:(题目中的要求的结果不同,结构体也不同,当要求求各个节点的和时,自然不用定义最大值与最小值)

struct Node
{
	int l;//储存左边界 // 
	int r;//储存右边界 //
	int sum;//储存各个节点的和 //
	int Max;//储存节点中最大数据 //
	int Min;//储存节点中最小数据//
}Tree[1000]//数组的大小根据提议而定//;
下面我们将构建线段树:

void Build(int o,int l,int r)
{
	//首先记录l和r的值 
	Tree[o].l = l;
	Tree[o].r = r;
	if (l == r)		//到达最底层,递归终止,就是上面那个图,最底层左边界与右边界是相等的,用这个判断是否到达最底层 
	{
		int t;
		scanf ("%d",&t);		//输入最底层数据,因为只有最底层的数据时输入的,其他的都是根据最底层推出来的 
		Tree[o].sum = Tree[o].Max = Tree[o].Min = t;		//更新节点数据 
		return;
	}
	int mid = (l+r) >> 1;		//找到中间节点 
	Build(o*2 , l , mid);		//递归建左子树 
	Build(o*2+1 , mid+1 , r);		//递归建右子树 
	PushUp(o);		//更新当前节点的值 
}
下面解释一下最后一步PushUp(o)更新节点值(注意不是节点更新)的函数:

void PushUp(int o)
{
	Tree[o].sum = Tree[o*2].sum + Tree[o*2+1].sum;
	Tree[o].Max = max(Tree[o*2].Max,Tree[o*2+1].Max);
	Tree[o].Min = min(Tree[o*2].Min,Tree[o*2+1].Min);
}(max,min时c++中两个常用的函数,头文件时#include<algorithm>加上using namespace std)
同过以上的过程便成功构建了线段树。

2.查询任意区间的最大值,最小值,区间和等题目要求输出的结果:

我们以查询x到y区间和为例:

int QuerySum(int o,int l,int r,int x,int y)		//查找x到y的和 
{
	if (l == x && r == y)		//如果恰好是当前节点,就返回 
	{
		return Tree[o].sum;
	}
	int mid = ( l + r ) / 2;
	if (mid >= y)		//全在左边 
		return QuerySum(o*2,l,mid,x,y);
	else if (x > mid)		//全在右边 
		return QuerySum(o*2+1,mid+1,r,x,y);
	else		//一半在左一半在右 
		return QuerySum(o*2,l,mid,x,mid) + QuerySum(o*2+1,mid+1,r,mid+1,y);
} //利用递归的方式一步步的查询//
3.将x借点更新为y节点:

void UpDate(int o,int l,int r,int x,int y)		//把x节点更新为y
{
	if (l == r)		//递归结束
	{
		Tree[o].Max = Tree[o].Min = Tree[o].sum = y;		//精确找到了节点,更新 
		return;
	}
	int mid = (l+r) / 2;		//找到中间位置
	if (x <= mid)
		UpDate(o*2,l,mid,x,y);		//找左子树 
	else
		UpDate(o*2+1,mid+1,r,x,y);		//找右子树 
	PushUp(o);		//更新当前节点 
}//利用递归一步步找到x节点//

下面看一个例题:

士兵杀敌(一)

时间限制:1000 ms  |  内存限制:65535 KB
难度:3
描述

南将军手下有N个士兵,分别编号1到N,这些士兵的杀敌数都是已知的。

小工是南将军手下的军师,南将军现在想知道第m号到第n号士兵的总杀敌数,请你帮助小工来回答南将军吧。

注意,南将军可能会问很多次问题。

输入
只有一组测试数据
第一行是两个整数N,M,其中N表示士兵的个数(1<N<1000000),M表示南将军询问的次数(1<M<100000)
随后的一行是N个整数,ai表示第i号士兵杀敌数目。(0<=ai<=100)
随后的M行每行有两个整数m,n,表示南将军想知道第m号到第n号士兵的总杀敌数(1<=m,n<=N)。
输出
对于每一个询问,输出总杀敌数
每个输出占一行
样例输入
5 2
1 2 3 4 5
1 3
2 4
样例输出
6
9

先看一下该题利用上述函数方式,解决代码:

#include<cstdio>
#include<algorithm>
using namespace std;
#define L o<<1
#define R (o<<1)|1
struct Node
{
	int l;//储存左边界 // 
	int r;//储存右边界 //
	int sum;//储存各个节点的和 //
}Tree[1000<<2];
void PushUp(int o)
{
	Tree[o].sum = Tree[o*2].sum + Tree[o*2+1].sum;
}
void Build(int o,int l,int r)
{
	//首先记录l和r的值 
	Tree[o].l = l;
	Tree[o].r = r;
	if (l == r)		//到达最底层,递归终止,就是上面那个图,最底层左边界与右边界是相等的,用这个判断是否到达最底层 
	{
		int t;
		scanf ("%d",&t);		//输入最底层数据,因为只有最底层的数据时输入的,其他的都是根据最底层推出来的 
		Tree[o].sum = t;		//更新节点数据 
		return;
	}
	int mid = (l+r) >> 1;		//找到中间节点 
	Build(o*2 , l , mid);		//递归建左子树 
	Build(o*2+1 , mid+1 , r);		//递归建右子树 
	PushUp(o);		//更新当前节点的值 
}
int QuerySum(int o,int l,int r,int x,int y)		//查找x到y的和 
{
	if (l == x && r == y)		//如果恰好是当前节点,就返回 
	{
		return Tree[o].sum;
	}
	int mid = (l + r) / 2;
	if (mid >= y)		//全在左边 
		return QuerySum(o*2,l,mid,x,y);
	else if (x > mid)		//全在右边 
		return QuerySum(o*2+1,mid+1,r,x,y);
	else		//一半在左一半在右 
		return QuerySum(o*2,l,mid,x,mid) + QuerySum(o*2+1,mid+1,r,mid+1,y);
} 
int main()
{
	int n,m;
	scanf ("%d%d",&n,&m);
	Build(1,1,n);
	int x,y;
	for(int i=1;i<=m;i++)
	{
		scanf("%d%d",&x,&y);
		printf ("%d\n",QuerySum(1,1,n,x,y));
	}
	
	return 0;
}
该题还有一种简单的方法,就是利用数组a[i]记录前i组数据的和,当给出区间x,y时用sum[y] - sum[x-1]即可,代码如下:

#include <stdio.h>  
int a[1123456];  
int sum[1123456];  //由于mian函数的储存空间有限,当数组比较大时,通常将数组定义在main函数的前面//
int main()  
{  
    int n,m;   
    int x,y; 
    scanf("%d %d", &n, &m);   
    for(int i = 1;i <= n;i++){  
        scanf("%d",&a[i]);  
        sum[i] = sum[i-1]+a[i];//sum[i]表示前i个数的总和//  
    }  
 
    for(int i = 0;i < m;i++)
	{  
        scanf("%d %d", &x, &y);  
        printf("%d\n", sum[y]-sum[x-1]);
    }  
  	return 0;  
}  

练习网站:https://cn.vjudge.net/contest/179845 

愿你一生清澈明朗,做你愿做之事,爱你愿爱之人!

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值