首页 > 其他分享 >线段树

线段树

时间:2023-02-24 01:12:08浏览次数:39  
标签:node int 线段 nodeTree st 区间 节点

还是区间求和问题

对于单点修改,区间求和问题,我们可以用树状数组很好地解决。但是如果需要对区间进行修改(统一加上一个值),然后区间求和这种问题,那么就需要线段树了。

线段树

线段树是一颗二叉树(近似于完全二叉树),线段树的每一个节点都对应一个区间[L,R],然后将该区间平分,分别作为左儿子和右耳子节点,直到区间变为一个点L == R,这时就变为一个叶子节点。

时间复杂度

显然线段树的高度为\(O(lgn)\)。对于区间查询操作,只需要从根节点开始查,不断划分下去,直到当前节点的范围满足当前的查询划分,最后总结起来就可以了。对于这一操作,可以粗略地证明时间复杂度是\(O(lgn)\)。

因为对于每一层的节点,如果连续的区间被选中,并且他们的父节点相同那么会合并为对父节点的查询,最终每一层最多有两个节点被选中。

然后就是对于区间的修改,这里需要用到懒操作,也是像查询区间一样,并不会细分到叶节点,然后只更新选中的区间对应的节点的值,并将修改值累计到该节点的懒操作缓存值中,只有从一个节点向其子节点继续操作的时候才将该懒操作的缓存值更新到选中的子节点。这样一来复杂度就和区间查询一样了,也为\(O(lgn)\)。

建树

可以用一个数组来保存线段树,在这个数组中左儿子下标为父节点下标的2倍,右儿子再加1,就和二叉堆一样。

并且数组的大小需要设置为区间范围的4倍。这个可以自己大概算一下,比如整个区间大小为\(2^n+1\),那么一共需要\(2^{n+2}+3\)个空间存储节点,非常接近4倍。

type SegTreeNode struct {
	left     int
	right    int
	sum      int
	delayVal int
}

type SegTree struct {
	nodeTree []SegTreeNode
	nums     []int
}

func leftChild(node int) int {
	return node << 1
}

func rightChild(node int) int {
	return (node << 1) + 1
}

func (st *SegTree) createST(nums []int) {
	st.nodeTree = make([]SegTreeNode, len(nums)*4)
	st.nums = nums
}

func (st *SegTree) buildST(left, right, node int) int {
	st.nodeTree[node].left = left
	st.nodeTree[node].right = right

	if left == right {
		// 叶子节点
		st.nodeTree[node].sum = st.nums[left]
		return st.nodeTree[node].sum
	}

	mid := (right-left+1)/2 + left

	st.nodeTree[node].sum = st.buildST(left, mid-1, leftChild(node)) + st.buildST(mid, right, rightChild(node))

	return st.nodeTree[node].sum
}

func main() {
	nums := []int{2, 4, 1, 5, 8}
	st := &SegTree{}
	st.createST(nums)
	st.buildST(0, len(nums)-1, 1)

}

查询区间

// 懒更新到下一层子节点
func (st *SegTree) pushDown(node int) {
	if st.nodeTree[node].delayVal != 0 {
		leftIntervalLen := st.nodeTree[leftChild(node)].right - st.nodeTree[leftChild(node)].left + 1
		st.nodeTree[leftChild(node)].sum += st.nodeTree[node].delayVal * leftIntervalLen
		st.nodeTree[leftChild(node)].delayVal += st.nodeTree[node].delayVal
		rightIntervalLen := st.nodeTree[rightChild(node)].right - st.nodeTree[rightChild(node)].left + 1
		st.nodeTree[rightChild(node)].sum += st.nodeTree[node].delayVal * rightIntervalLen
		st.nodeTree[rightChild(node)].delayVal += st.nodeTree[node].delayVal

		st.nodeTree[node].delayVal = 0
	}
}

func (st *SegTree) intervalSum(l, r, node int) int {
	if st.nodeTree[node].left == l && st.nodeTree[node].right == r {
		return st.nodeTree[node].sum
	}

	st.pushDown(node)

	mid := (st.nodeTree[node].right-st.nodeTree[node].left+1)/2 + st.nodeTree[node].left

	if mid <= l {
		return st.intervalSum(l, r, rightChild(node))
	} else if mid > r {
		return st.intervalSum(l, r, leftChild(node))
	} else {
		return st.intervalSum(l, mid-1, leftChild(node)) + st.intervalSum(mid, r, rightChild(node))
	}
}

更新区间


func (st *SegTree) intervalAdd(l, r, val, node int) int {
	if st.nodeTree[node].left == l && st.nodeTree[node].right == r {
		st.nodeTree[node].sum += val * (r - l + 1)
		st.nodeTree[node].delayVal = val
		return val * (r - l + 1)
	}

	st.pushDown(node)

	mid := (st.nodeTree[node].right-st.nodeTree[node].left+1)/2 + st.nodeTree[node].left

	var total int
	if mid <= l {
		total = st.intervalAdd(l, r, val, rightChild(node))
	} else if mid > r {
		total = st.intervalAdd(l, r, val, leftChild(node))
	} else {
		total = st.intervalAdd(l, mid-1, val, leftChild(node)) + st.intervalAdd(mid, r, val, rightChild(node))
	}
	st.nodeTree[node].sum += total
	return total
}

参开资料

标签:node,int,线段,nodeTree,st,区间,节点
From: https://www.cnblogs.com/HachikoT/p/17149988.html

相关文章

  • 线段树学习
    参考资料:《算法竞赛入门到进阶》、左神算法课线段树示意图,图片摘自《算法竞赛入门到进阶》解决区间更新、区间查询、区间累加问题基于原数组的4倍数组长度建立sum......
  • 【YBT2023寒假Day12 B】仰望星空(DP)(线段树)(笛卡尔树)
    仰望星空题目链接:YBT2023寒假Day12B题目大意有一个n*n的网格,第i列下面的ai个点都是障碍。然后又一些不是障碍的地方有特殊点,删掉它有费用。要你用最小费用使得......
  • leetcode 307. 区域和检索 - 数组可修改 前缀和 | 线段树
    前缀和查询是O(1)更新是O(n)classNumArray{public:vector<int>sum;vector<int>nums;NumArray(vector<int>&nums){this->nums=nums;sum......
  • H - 线段树 1【GDUT_22级寒假训练专题五】
    H-线段树1原题链接题意区间修改区间查询线段树简介线段树是一颗二叉树,他能通过树的分支将一块区间划分为数个单元区间,每个单元区间对应线段树中的一个叶结点。如......
  • 【ZJOI2019】线段树
    【ZJOI2019】线段树Description九条可怜是一个喜欢数据结构的女孩子,在常见的数据结构中,可怜最喜欢的就是线段树。线段树的核心是懒标记,下面是一个带懒标记的线段树的伪......
  • hihoCoder 1078 : 线段树的区间修改
    #1078:线段树的区间修改10000ms1000ms256MB描述对于小Ho表现出的对线段树的理解,小Hi表示挺满意的,但是满意就够了么?于是小Hi将问题改了改,又出给了小Ho:假设货架上......
  • 线段树板子C++
    structnode{intl,r,sum,lazy;node*lson,*rson;node(){l=r=sum=lazy=0;lson=rson......
  • 区间和线段树的各种操作
    这篇文章我们来讲一下线段树线段树,适用于对一个数列区间进行操作,可以求这段数列\(i\)到\(j\)的和、乘积、最大值、最小值等等等等,因此线段树有十分多的变种问题提出如......
  • 线段树
    前言线段树经常用来维护区间信息,支持单点修改、区间修改、单点查询、区间查询等操作。其中单点修改/查询其实可以类似地看成区间长度为\(1\)的区间。而上面的操作时间复......
  • 线段树
    权值线段树线段树维护的区间是一个值域范围,而不是一段下标的区间。P1637三元上升子序列我们可以考虑中间的数字\(x\),统计前面比\(x\)小的数字个数\(a_1\),统计后......