首页 关于 友链

线段树入门详解

swwind

简介

线段树其实就是一个超级简单的数据结构。就是把一个序列搞出若干个区间,在询问和修改时改一下区间就行了。

比方说某节点表示 \( [1, n] \) 的区间,那么该节点的左子树就表示 \( [1, (n+1)/2] \) 的区间,右子树就表示 \( [(n+1)/2+1, n] \) 的区间。然后每个节点维护一下该区间内的目标对象(sum、max等)就行了,这样可以在 \( O(n^2) \) 的时间内建树,在 \( O(\log(n)) \) 的时间内完成查询操作。

注意到线段树是颗二叉树,所以每个节点的左儿子可以表示为 x*2 (位运算简化为 x<<1),右儿子为 x*2+1(x<<1|1)。注意在查询的时候如果要修改的刚好就是该节点的区间,那么可以为这个节点打一个flag(俗称Lazy标记),而不必把他下面的节点全部更新(不然是 \( O(n) \) 的了)。

模板

下面就是模板了,但是希望大家先不看模板打一遍,再与模板校对并优化

先是每个节点的结构体:

struct node{
	int l, r, sum, mx, lazy; // l, r: 左右边界,sum、mx: 最大值 lazy只是个标记
}tr[N<<2]; // 节点要开长度的四倍大小

建树:

void build(int x, int l, int r){
	tr[x].l = l, tr[x].r = r;
	if(l == r){
		tr[x].sum = tr[x].mx = a[l];
		return;
	}
	int mid = tr[x].l + tr[x].r >> 1;
	build(x<<1, l, mid);
	build(x<<1|1, mid+1, r);
	push_up(x); // 更新x的值
}

然后是push_up的内容:

void push_up(int x){
	tr[x].sum = tr[x<<1].sum + tr[x<<1|1].sum + a[x]; // 更新sum
	tr[x].mx = max(tr[x<<1].mx, tr[x<<1|1].mx); // 更新max
}

接着是查询:

int asksum(int x, int l, int r){
	push_down(x); // 这步后来会讲到,下传标记
	if(tr[x].l == l && tr[x].r == r) return tr[x].sum;
	int mid = tr[x].l + tr[x].r >> 1;
	if(r <= mid) return asksum(x<<1, l, r); // 若果询问的区间明显在节点区间的左半边,那么去问节点的左子树
	if(l > mid) return asksum(x<<1|1, l, r); // 如果在右边就去问右子树
	return asksum(x<<1, l, mid) + asksum(x<<1|1, mid+1, r); // 不然两个都要问
}
int askmax(int x, int l, int r){ // 这个和上面一个基本一样
	push_down(x);
	if(tr[x].l == l && tr[x].r == r) return tr[x].mx;
	int mid = tr[x].l + tr[x].r >> 1;
	if(r <= mid) return askmax(x<<1, l, r);
	if(l > mid) return askmax(x<<1|1, l, r);
	return max(askmax(x<<1, l, mid), askmax(x<<1|1, mid+1, r));
}

然后是区间加的操作:

void update(int x, int l, int r, int v){
	if(tr[x].l == l && tr[x].r == r){
		tr[x].lazy += val; // 这标记就是说,他的儿子都要加这么多,但是懒得算下去了
		tr[x].sum += val*(tr[x].r - tr[x].l + 1);
		tr[x].mx += val;
		return;
	}
	int mid = tr[x].l + tr[x].r >> 1;
	if(r <= mid) update(x<<1, l, r, val);
	else if(l > mid) update(x<<1|1, l, r, val);
	else update(x<<1, l, mid, val), update(x<<1|1, mid+1, r, val);
	push_up(x); // 更新x的值
}

然后就是标记下传啦!

void push_down(int x){
	if(!tr[x].lazy) return; // 没有标记就返回
	if(tr[x].l){ // 有左儿子
		tr[tr[x].l].lazy += tr[x].lazy;
		tr[tr[x].l].sum += tr[x].lazy*(tr[x].r - tr[x].l + 1);
		tr[tr[x].l].mx += tr[x].lazy;
	}
	if(tr[x].r){ // 有右儿子
		tr[tr[x].r].lazy += tr[x].lazy;
		tr[tr[x].r].sum += tr[x].lazy*(tr[x].r - tr[x].l + 1);
		tr[tr[x].r].mx += tr[x].lazy;
	}
	tr[x].lazy = 0;
}

最后

看完了之后就有一大波水题在等着你!

就算看不懂照题解多打几遍慢慢就懂了的!

刷题才是王道!

GL&HF

Copyright © 2017-2025 swwind. All rights reserved
Except where otherwise noted, content on this blog is licensed under CC-BY 2.0