模板:线段树

文章目录

  • 引言
  • 思想
  • 模板
    • 建树
    • 单点修改 / 查询
    • 区间修改/查询
  • 总结
  • 练习


引言

有一类题目:要求在区间上维护信息,比如带修改区间求和问题。考虑到枚举求和肯定会超时,我们可以通过一些数据结构来维护信息,例如线段树。
它功能强大,支持区间求和,区间求最值、区间修改等一系列操作。

思想

线段树,是一种二叉搜索树(即每个结点最多有两棵子树,通常叫做左右子树)。
它将一段区间划分为若干单位区间,每一个节点都储存着一段区间[L,R]的信息,其中叶子节点L=R。它的大致思想是:将区间[1,n]平均分成2个小区间,再将小区间平均分成2个更小区间…以此类推,直到区间长度变为1无法分成两个更小的区间。通过对这些小区间进行修改、查询,来实现对大区间的修改、查询。
用线段树维护的问题必须满足可以用两个小区间的信息合并成大区间信息,否则是不可能将大问题划分成子问题来解决的。

下图就是一棵[1,10]的线段树的分解过程。(相同颜色的节点在同一层)
可以证明,线段树的最大深度不超过log2n
在这里插入图片描述

模板

整体来说都是在贯彻递归二分分治的思想。注意k结点的两个孩子编号应分别为2k 和 2k+1
注意:关于线段树的数组要开到4n(最差时)!

建树

思想:常见的做法是遍历整棵线段树,设当前遍历到的节点所代表的区间为[l,r],设中点mid=(l+r)/2,则左子树区间[l,mid],右子树的区间为[mid+1,r],注意要递归到线段树的叶节点才结束,并给每一个节点赋值,叶子节点直接赋值,非叶子节点信息由左右子树信息合并而来。

void build(int k,int l,int r){if(l==r){sum[k]=f[l];return;}int mid=(l+r)>>1;build(2*k,l,mid);build(2*k+1,mid+1,r);sum[k]=sum[2*k]+sum[2*k+1];
}

单点修改 / 查询

用树状数组不好吗
思想:如果当前遍历到目标节点,就返回节点信息。如果不是,设查询位置为x,当前结点区间范围为了[l,r],中点为mid,若x≤mid,则x在左子树的区间,递归当前结点的左孩子;否则x在右子树的区间,递归它的右孩子。

int dotquery(int k,int l,int r,int x){if(l==x)return sum[k];int mid=(l+r) /2;int res=0;if(x<=mid)	return dotquery(2*k,l,mid,x);else return dotquery(2*k+1,mid+1,r,x);
} void dotchange(int k,int l,int r,int x,int v){if(l==r){sum[k]+=v;return;}int mid=(l+r) /2;if(x<=mid) dotchange(2*k,l,mid,x,v);else dotchange(2*k+1,mid+1,r,x,v);sum[k]=sum[2*k]+sum[2*k+1];
}

区间修改/查询

区间查询:
设待查询区间为[x,y],如果当前遍历到的区间被查询区间包含,就直接返回节点信息。如果不是,设当前结点区间范围为[l,r]中点为mid,如果x≤mid,则待查询区间与[l,mid]有交集,递归[l,mid];如果y>mid,则待查询区间与[mid+1,r]有交集,递归[mid+1,r],最后将返回的信息合并就是区间[x,y]的答案。
区间修改
延续区间查询的思想,我们可以找出所有信息发生改变的节点,可现在有一核心问题:我们并不能直接在这些节点上修改,若全部修改,复杂度难以承受。所以,区间修改的关键思路是:只修改对查询有用的点。为此,我们引入“懒标记"的概念。
懒标记:若遍历到的节点带有懒标记,则立即修改当前节点的信息,并把标记下放到左右儿子,然而左右子树的信息不变动,只有遍历到左/右节点时才修改其信息

void Add(int k,int l,int r,int v){add[k]+=v;sum[k]+=(ll)(r-l+1)*v;return;
}
void pushdown(int k,int l,int r,int mid){if(add[k]==0) return;Add(2*k,l,mid,add[k]);Add(2*k+1,mid+1,r,add[k]);add[k]=0;
}
void longchange(int k,int l,int r,int x,int y,int v){if(x<=l&&r<=y){Add(k,l,r,v);return;}int mid=(l+r)>>1;pushdown(k,l,r,mid);if(x<=mid) longchange(2*k,l,mid,x,y,v);if(y>=mid+1) longchange(2*k+1,mid+1,r,x,y,v);sum[k]=sum[2*k]+sum[2*k+1];
}
ll longquery(int k,int l,int r,int x,int y){if(x<=l&&r<=y) return sum[k];int mid=(l+r)>>1;pushdown(k,l,r,mid);ll res=0;if(x<=mid) res+=longquery(2*k,l,mid,x,y);if(y>=mid+1) res+=longquery(2*k+1,mid+1,r,x,y);return res;
}

总结

线段树是很常用强大的数据结构,其功能也远不止求最值和加和这么简单,要想真正掌握,还是要背模板 理解其深层的运作机制

练习

洛谷题单传送门

欢迎各位dl点赞评论,谢谢 收听 拜读观看! awa

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/318757.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Docker最全教程——从理论到实战(三)

容器是应用走向云端之后必然的发展趋势&#xff0c;因此笔者非常乐于和大家分享我们这段时间对容器的理解、心得和实践。本篇教程持续编写了2个星期左右&#xff0c;只是为了大家更好地了解、理解和消化这个技术&#xff0c;能够搭上这波车。你可以关注我们的公众号“magiccode…

2021牛客暑期多校训练营4 H-Convolution(数学)

H-Convolution x⨂yxygcd⁡(x,y)x\bigotimes y\frac{xy}{\gcd(x,y)}x⨂ygcd(x,y)xy​ 下面大佬题解推&#x1f981; Ultraman-Ace题解 #include<bits/stdc.h> using namespace std; using lllong long; template <class Tint> T rd() {T res0;T fg1;char chgetcha…

牛客题霸 [ 排序] C++题解/答案

题目描述 给定一个数组&#xff0c;请你编写一个函数&#xff0c;返回该数组排序后的形式。 题意&#xff1a; 排序的方式有很多 二分呀&#xff0c;桶排呀归并等等 stl里的sort快速排序方便好用 题解&#xff1a; class Solution { public:/*** 代码中的类名、方法名、参…

ARC106E-Medals【hall定理,高维前缀和】

正题 题目链接:https://atcoder.jp/contests/arc106/tasks/arc106_e 题目大意 nnn个员工&#xff0c;第iii个在[1,Ai][1,A_i][1,Ai​]工作&#xff0c;[Ai1,2Ai][A_i1,2\times A_{i}][Ai​1,2Ai​]休息&#xff0c;[2Ai1,3Ai][2\times A_i1,3\times A_i][2Ai​1,3Ai​]工作…以…

高并发、低延迟之C#玩转CPU高速缓存(附示例)

写在前面好久没有写博客了&#xff0c;一直在不断地探索响应式DDD&#xff0c;又get到了很多新知识&#xff0c;解惑了很多老问题&#xff0c;最近读了Martin Fowler大师一篇非常精彩的博客The LMAX Architecture&#xff0c;里面有一个术语Mechanical Sympathy&#xff0c;姑且…

RMQ问题:与众不同(st表的高端应用)

解析 预处理 用pre[i]表示以i结尾的最长完美序列起始点&#xff0c;用last[i]表示数字i最后出现的位置 那么可以得到递推式&#xff1a; pre[i]max(pre[i-1],last[x[i]]1);也就是说这个pre要么是受前一位一样的限制&#xff0c;要么是受自己的限制 用f[i]表示以i结尾的最长完…

codeforces1554 E. You(思维+数学+转化)

E. You 首先我们假设每个节点有个点权&#xff0c;点权是儿子的数量。 一个点可以和他的父亲所要权值1&#xff08;先删去自己&#xff0c;再删去父亲&#xff09;&#xff0c;并且此操作不可逆。于是dfs从叶子节点一次递推即可得出某个kkk是否能作为gcd⁡\gcdgcd。 由于上述…

牛客题霸 [括号生成] C++题解/答案

牛客题霸 [括号生成] C题解/答案 题意&#xff1a; 给出n对括号&#xff0c;请编写一个函数来生成所有的由n对括号组成的合法组合。 例如&#xff0c;给出n3&#xff0c;解集为&#xff1a; “((()))”, “(()())”, “(())()”, “()()()”, “()(())”, 题解&#xff1a; …

P5404-[CTS2019]重复【KMP,dp】

正题 题目链接:https://www.luogu.com.cn/problem/P5404 题目大意 给出一个字符串SSS&#xff0c;然后求有多少个长度为mmm的串TTT满足。无限多个串TTT拼接起来后能找出一个长度和SSS相等的子串字典序比SSS小。 1≤∣S∣,m≤20001\leq |S|,m\leq 20001≤∣S∣,m≤2000 解题思…

Docker最全教程——从理论到实战(四)

容器是应用走向云端之后必然的发展趋势&#xff0c;因此笔者非常乐于和大家分享我们这段时间对容器的理解、心得和实践。本篇教程持续编写了2个星期左右并且一直在完善、补充具体的细节和实践&#xff0c;预计全部完成需要1到2个月的时间。由于编写的过程中极其费时&#xff0c…

模板:高精

想必大家一定都深受高精所害 python党&#xff1a;啥叫高精&#xff1f; 这里送上一种高精板子 写法 转载自巨神wyx的博客 #include <bits/stdc.h> using namespace std;const int maxn 1000;struct bign {int d[maxn], len;void clean() {while (len > 1 &&…

牛客题霸 [最长公共子序列] C++题解/答案

牛客题霸 [最长公共子序列] C题解/答案 题目描述 给定两个字符串str1和str2&#xff0c;输出连个字符串的最长公共子序列。如过最长公共子序列为空&#xff0c;则输出-1。 题解&#xff1a; dp经典问题 代码&#xff1a; class Solution { public:/*** longest common su…

2021牛客暑期多校训练营4 G-Product(组合意义+容斥原理)

G-Product Spy_Savior题解 Ultraman-Ace题解 组合意义容斥原理 转化后题目关键要求有DnkDnkDnk个小球放在nnn个盒子中&#xff0c;每个盒子至少有k个小球的方案数。 题目的等价式子为 ∑∑i1naiDnk[ai≥k]D!∏i1nai!\sum_{\sum_{i1}^na_iDnk[a_i\ge k]}\frac{D!}{\prod_{i1}^…

P6122-[NEERC2016]Mole Tunnels【模拟费用流】

正题 题目链接:https://www.luogu.com.cn/problem/P6122 题目大意 给出nnn个点的一棵满二叉树&#xff0c;每个点有容量cic_ici​&#xff0c;mmm次从pip_ipi​处加一只仓鼠然后求每只仓鼠都到一个点的最短路径长度和。 1≤n≤1051\leq n\leq 10^51≤n≤105 解题思路 模拟费…

牛客题霸 [矩阵查找] C++题解/答案

牛客题霸 [矩阵查找] C题解/答案 题目描述 请写出一个高效的在m*n矩阵中判断目标值是否存在的算法&#xff0c;矩阵具有如下特征&#xff1a; 每一行的数字都从左到右排序 每一行的第一个数字都比上一行最后一个数字大 例如&#xff1a; 对于下面的矩阵&#xff1a; [ [1, 3,…

AtCoder Beginner Contest 211 E - Red Polyomino(暴力+状态记录)

E - Red Polyomino 暴力&#xff0c;对于状态的记录考虑下面用set<vector<string>> mp #include<bits/stdc.h> using namespace std; using lllong long; template <class Tint> T rd() {T res0;T fg1;char chgetchar();while(!isdigit(ch)) {if(ch-…

CF960G-Bandit Blues【第一类斯特林数,分治,NTT】

正题 题目链接:https://www.luogu.com.cn/problem/CF960G 题目大意 求有多少个长度为nnn的排列&#xff0c;使得有AAA个前缀最大值和BBB个后缀最大值。 0≤n,A,B≤1050\leq n,A,B\leq 10^50≤n,A,B≤105 解题思路 显然的是把最大的数两边然后左边的是前缀最大值&#xff0c;…

动态规划:openjudge 2.6-3532 最大上升子序列和 解题心得

传送门 题目描述 一个数的序列bi&#xff0c;当b1 < b2 < … < bS的时候&#xff0c;我们称这个序列是上升的。对于给定的一个序列(a1, a2, …,aN)&#xff0c;我们可以得到一些上升的子序列(ai1, ai2, …, aiK)&#xff0c;这里1 < i1 < i2 < … < iK…

边缘化搭建 DotNet Core 2.1 自动化发布和部署(下)

写在前面本篇文章是上一篇边缘化搭建 DotNet Core 2.1 自动化发布和部署(上)的后续操作&#xff0c;本文主要讲解如何开启Docker Remote API&#xff0c;开启Remote API后的权限安全问题。配置Jenkins构建项目&#xff0c;并在云服务器上构建成功。废话不多说&#xff0c;我们一…

牛客题霸 [二进制中1的个数] C++题解/答案

牛客题霸 [二进制中1的个数] C题解/答案 题目描述 输入一个整数&#xff0c;输出该数32位二进制表示中1的个数。其中负数用补码表示。 题解&#xff1a; 判断1的个数 x&(-x)2^k 有点类似于树状数组中lowbit的操作 代码&#xff1a; class Solution { public:int Num…