首页 > 其他分享 >Segment tree Beats!

Segment tree Beats!

时间:2023-09-15 18:12:00浏览次数:39  
标签:log 标记 最大值 tree 最小值 Beats 区间 Segment 我们

前言

Segment Beats 可以用来解决处理区间最值,区间历史值的问题。

不保证这些题都实现过

区间最值操作

HDU5306. Gorgeous Sequence
给出一个长度为 \(n(n\le 10^6)\) 的序列 \(A\) 和 \(m(m\leq 10^6)\) 次操作,每次操作为以下三种类型之一:

  1. 给出 \(l,r,k\),对于所有 \(i\in[l,r]\),将 \(A_i\) 变成 \(\min(A_i,k)\);
  2. 给出 \(l,r\),求序列 \(A\) 区间 \([l,r]\) 的最大值;
  3. 给出 \(l,r\),求 \(\sum_{i=l}^{r}A_i\)。

我们考虑这样一个做法,维护区间最大值 \(mx\) ,最小值 \(mn\),和 \(sum\) ,然后对于修改,如果 \(mx\le k\) 就直接退出,\(mn\ge k\) 就区间赋值,否则就递归,这是一个 \(O(n^2\log n)\) 的做法,很明显过不了。

我们换一种方式,记录区间的最大值 \(mx\),最大值个数 \(cnt\),和区间严格次大值 \(se\)。那么对于一次修改,有三种情况。

  • \(mx\le k\),此时直接退出。
  • \(se<k\le mx\),对最大值打上修改 \(tag\) ,这里用赋值或者加法标记都可以,根据 \(cnt\) 修改 \(mx,sum\),退出。
  • \(k\le se\),在这种情况下,我们递归处理。

这个做法看起来暴力,但是可以证明这个做法的时间复杂度是 \(O((n+m)\log n)\) 的。

我们令这个节点的最大值和其父亲最大值不同的节点为关键点。

设一次操作中的关键点点数为 \(A\),则访问的总点数是 \(O(A\log n)\) 的,我们花费的时间为 \(O(A\log n)\),而关键点个数减少了 \(A\)。

刚开始的时候有 \(O(n)\) 个关键点,所以总时间复杂度是 \(O((n+m)\log n)\) 的。

这种划分最值与非最值,然后转化成加减修改的做法是区间最值操作的核心,请牢牢记住。

例题二
给出一个长度为 \(n(n\le 10^6)\) 的序列 \(A\) 和 \(m(m\leq 10^6)\) 次操作,每次操作为以下四种类型之一:

  1. 给出 \(l,r,k\),对于所有 \(i\in[l,r]\),将 \(A_i\) 变成 \(\min(A_i,k)\);
  2. 给出 \(l,r,v\),对于所有 \(i\in[l,r],A_i\gets A_i+v\) 。
  3. 给出 \(l,r\),求序列 \(A\) 区间 \([l,r]\) 的最大值;
  4. 给出 \(l,r\),求 \(\sum_{i=l}^{r}A_i\)。

这道题我们沿用上一题的思路,将维护的元素分成最大值和非最大值两类,分开维护。

对于这个做法,论文中给出了一个 \(O(n\log n+m\log^2n)\) 的证明。

同样,我们令这个节点的最大值和其父亲最大值不同的节点为关键点。

设每次 \(1\) 操作的减少了 \(A\) 个关键点,则代价为 \(O(A\log n)\),然后每一次 \(2\) 操作,最多令线段树上修改的 \(O(\log n)\) 个节点成为关键点,所以总关键点数是 \((n+m\log n)\) 的,时间复杂度为 \(O(n\log n+m\log^2 n)\) 。

UOJ#515. 【UR #19】前进四
给出一个长度为 \(n(n\leq 10^6)\) 的序列 \(A\) 和 \(q(q\leq 10^6)\) 次操作,每次操作为以下两种类型之一:

  1. 给出 \(x,v\),将 \(A_x\) 变成 \(v\);
  2. 给出 \(x\),求 \(A_x,A_{x+1},\cdots,A_n\) 的不同后缀最小值个数。

考虑到询问的是后缀,我们可以对下标从后往前扫描线,维护每一个时刻 \(A_x\) 的值,如果一个时刻的 \(A_x\) 小于之前这个时刻的历史最小值,那么我们就令这个时刻的答案加 \(1\),转化成区间取 \(\min\) ,如果被修改到就加 \(1\) 的问题,这个很好维护。

CF1290E Cartesian Tree
给你一个排列,对于每一个 \(k\in[1,n]\),求出只保留 \([1,k]\) 时,建大根笛卡尔树所有节点的子树大小之和。

先考虑对于 \(k\) 等于 \(n\) 怎么做。

对于下标 \(i\),记 \(i\) 左边第一个大于它的下标为 \(lst_i\),下一个为 \(nxt_i\),则下标 \(i\) 的子树大小为 \(nxt_i-lst_i-1\)。

然后大小之和即为 \(\sum nxt_i-\sum lst_i-n\),\(nxt\) 与 \(lst\) 的维护方法是相同的,我们现在只讨论 \(nxt\) 的维护方法。

现在我们令 \(k\) 从小到大增加。

我们每次增加了一个比之前的所有数都大的数,所以其左侧的点的 \(nxt\) 都应该和 \(k\) 的下标取 \(\min\) 。

然后我们发现,对于空着的位置,我们是不计入下标的,我们其实是在插入,然后右侧的点的下标都会加 \(1\),同时 \(nxt\) 也会加 \(1\) 。

我们就把这个题转化成了区间取 \(\min\),区间加,单点修改,全局求和的问题。

Picks loves segment tree V
维护 \(A,B\) 两个数组,对 \(A\) 数组区间取 \(\max\),区间加,询问区间 \(A_i-B_i\) 的最大值和最小值。

我们将区间中的下标分为四种:\(A,B\) 同时为最小值,\(A\) 为最小值但 \(B\) 不是,\(A\) 不是最小值但 \(B\) 是,\(A,B\) 都不是最小值。

然后对于这四种情况分别处理,维护一下区间 \(A,B\) 最小值,次小值,四类的 \(A-B\) 最值。

我们发现这种做法能拓展到 \(K\) 个数列的情况,时间复杂度 \(O(2^Km\log^2 n)\),空间复杂度 \(O(2^K n)\)。

Dzy loves segment tree
区间取 \(\min\),区间加,询问区间 \(\gcd\)。

我们同样分为最大值和非最大值,对非最大值差分,加减变成对差分部分做单点加,对非差分部分区间加,分开做就可以了。

复杂度 \(O(n\log^3 n)\)。

CF997E Good Subsegments
给出长度为 \(n\) 的一个排列。

定义一个集合是连续的,当且仅当 \(\max S-\min S+1=|S|\)

\(q\) 次询问某个区间 \([l,r]\) 有多少个子区间是连续的。

这个大家可以自己思考一下。

历史最值操作

在完成接下来的例题前,请大家记住一点,我们的懒标记本质是若干个标记形成的队列的等价标记。

如我们的标记本来是 \(q_1,q_2,q_3\ldots q_k\),在平时我们能将其合并成一个等价的标记 \(q\)。

但在历史值问题中,我们不仅要维护这个等价的标记,还要维护这个标记队列的历史最值的标记 \(q'\)。

在标记叠加的时候,我们可以从这方面来思考新的标记对原合并的等价标记和在此基础上产生的历史标记带来的影响。

拉一些 cmd 博客里的话过来:

在线段树标记的下推机制中,某个点存有标记,表示整一棵子树在标记存在的时间内都未曾更新。

于是,问题的核心就在于分析单个节点上停留的标记的影响。

在非历史值问题中,我们只关注该节点“当下”的标记是什么,所以我们会直接将标记合并起来。

但在历史值问题中,我们不仅要考虑该节点现在的标记合并结果,还要考虑历史上推来的(按时间为序的)每个标记的依次作用。

为了便于理解,我们不合并标记,假设每个节点上有一个队列(按照时间先进先出),放着所有曾经推过来的标记。

下推标记时,将该节点上所有的标记推到两个儿子处,并清空队列。

于是,我们寻找一种方法 概括一个队列的标记对当前节点的影响

CPU 监控

区间赋值,区间加,区间最大值,区间历史最大值的最大值。

我们对于每个节点,维护最大值 \(v\),加法标记 \(add\),赋值标记 \(tag\),历史最大值 \(hv\),历史最大加法标记 \(hadd\),历史最大赋值标记 \(htag\)。

我们发现对于一个赋值操作后的加法操作,我们是可以将其转化成赋值操作的。

对于 \(u\to v\) 标记的下传 ,分为 \(add\) 标记和 \(tag\) 标记下传。

  • 对于 \(add\) 标记,\(v.hv=\max(v.hv,v.v+u.hadd),v.hadd=\max(v.hadd,v.add+u.hadd)\)。
  • 对于 \(tag\) 标记,\(v.hv=\max(v.hv,u.htag),v.htag=\max(v.htag,u.htag)\)。

可以自己理解一下,\(add,tag,v\) 应该都会维护,这里就不说了。

GSS2 - Can you answer these queries II
对于每一个数只算一次,我们直接将区间 \([lst_v,i]\) 加上 \(v\) 就行了。

然后我们发现,我们询问的 \([l,r]\) 的最大值,实际上是 \([l,l\sim r],[l+1,l+1\sim r],\ldots [r,r]\) 的历史最大值。

然后问题转化成区间加,区间历史最大值的最大值。

线段树 3
区间加,区间取 \(\min\) ,区间求和,区间最大值,区间历史最大值的最大值。

这个题将区间最值和历史最值联系起来了。

我们发现对于一个线段树节点整体操作的时候,其中的每个元素的相对大小关系是不变的。

比如在其中 \(a_i<a_j\),不会因为整体操作后就变成 \(a_i>a_j\)。

所以我们可以将最大值和非最大值分开维护,然后这个题就很容易解决了。

UOJ#164. 【清华集训2015】V
区间加,区间变成 \(\max(A_i-k,0)\),区间赋值,询问单点值和历史最大值。

吐槽:这个东西可以直接分成最大值和非最大值讨论做的,不明白为什么论文里要讲标记合并的做法,upd:好像用标记是一个 $\log $ 的

我们考虑维护一个标记 \((add,v)\) 表示将区间的数都加上 \(add\),然后对 \(v\) 取\(\max\),我们发现他对于区间的数是一个前面相同,后面斜率为 \(1\) 的形式,下传标记是将对应位置上的数取 \(\max\),然后大概长这样。(图片来源:灵梦的博客

对于三种修改,分别就是 \((v,-\infty),(-k,0),(-\infty,v)\)。

现在我们来考虑如何合并标记,对于两个标记 \((a,b),(c,d)\),我们将其合并就变成了 \((c,\max(b+c,d))\),这个应该是很好理解的,然后我们就把这个题做完了。

UOJ#169. 【UR #11】元旦老人与数列
区间加,区间取 \(\max\),区间最小值,区间历史最大值的最小值。

我们发现这个如果还是像上一个题一样维护是不大行的,因为可能会因为标记是最小值而操作是取 \(\max\) 成为 \(n\) 段函数,就像下图一样。

我们还是考虑将数分成最小值和非最小值考虑,因为对同一个区间整体修改元素的相对大小不变,所以就很好维护了。

我们维护区间最小值,区间严格次小值,区间历史最大值的最小值,区间最小值加减标记,区间最小值历史最大加减标记,区间非最小值加减标记,区间非最小值的历史最大加减标记就可以了。

维护历史最值和
区间 \(A_i\) 加 \(v\),\(B_i\) 是 \(A_i\) 的历史最小值,区间 \(B_i\) 求和。

我们发现上面的方法都无法沿用,我们考虑将其转化成区间最值操作,我们定义一个辅助数组 \(C\),\(\forall i\in[1,n],C_i=A_i-B_i\)。

然后对于 \(A_i\gets A_i+v\),讨论一下。

  • \(A_i+v\ge B_i\),此时 \(B_i\) 不变,\(C_i\gets C_i+v\)
  • \(A_i+v<B_i\),此时 \(B_i\gets A_i+v,C_i=0\)

所以,总结一下,修改就是就是 \(\forall i\in[l,r],C_i=\max(C_i+k,0)\)。

询问时我们将 \(A\) 的区间和减去 \(C\) 的区间和就可以了。

标签:log,标记,最大值,tree,最小值,Beats,区间,Segment,我们
From: https://www.cnblogs.com/nich666/p/17705651.html

相关文章

  • STL(12) RBTREE 红黑树
    目录红黑树的基本原理基本要求变色和旋转rbtree源码G2.9示例2.94.9treenode的构造关联式容器:查找快,插入快STL中的主要代表:红黑树,hashtable红黑树的基本原理单个结点来看,左孩子小于根节点,右孩子大于根节点(二叉搜索树)红黑树是什么,有什么意义:排序二叉树有不平衡的问题,可能左......
  • AtCoder Grand Contest 058 F Authentic Tree DP
    洛谷传送门AtCoder传送门人生中第一道AtCoder问号题。设\(P=998244353\)。注意到\(f(T)\)的定义式中,\(\frac{1}{n}\)大概是启示我们转成概率去做。发现若把\(\frac{1}{n}\)换成\(\frac{1}{n-1}\)答案就是\(1\),所以\(\frac{1}{n}\)大概是要转成点数之类的。......
  • AGC058 F Authentic Tree DP
    一道问号题,AT赛场上没人通过。其实是联考题这道题十分有意思,做法很简单但是要想到是极其困难的。考场上我也拿着推了很久猜测这个式子的组合意义,擦到了正解的一些边。然而正解的思想还是太反直觉了。首先题目中的式子实际上是让我们对树上的边建一颗笛卡尔树,然后计算笛卡尔树所......
  • TreeView的基本使用,以及和TableView的区别
    Qt中的QTreeView是一个用于显示树形数据的强大控件,通常用于显示层次结构数据。以下是使用QTreeView的基本步骤:创建一个QTreeView实例:在你的主窗口或其他窗口部件中创建一个QTreeView实例:QTreeView*treeView=newQTreeView(this);创建一个数据模型:QTreeView需要一个数......
  • 1135 Is It A Red-Black Tree
    题目:Thereisakindofbalancedbinarysearchtreenamed red-blacktree inthedatastructure.Ithasthefollowing5properties:(1)Everynodeiseitherredorblack.(2)Therootisblack.(3)Everyleaf(NULL)isblack.(4)Ifanodeisred,thenbot......
  • Java树形菜单_轻量级js树形插件_jsTree树形插件
    //插件效果//代码<!DOCTYPEhtml><html><head><title>JS轻量级树形插件</title><metacharset="utf-8"><linkrel="stylesheet"href="https://cdnjs.cloudflare.com/ajax/libs/jstree/3.2.1/themes/def......
  • 界面控件DevExpress WPF TreeMap,轻松可视化复杂的分层结构数据!
    DevExpressWPF TreeMap控件允许用户使用嵌套的矩形块可视化复杂的平面或分层结构数据。P.S:DevExpressWPF拥有120+个控件和库,将帮助您交付满足甚至超出企业需求的高性能业务应用程序。通过DevExpressWPF能创建有着强大互动功能的XAML基础应用程序,这些应用程序专注于当代客户的......
  • 计算机视觉算法中的GrabCut图像分割(GrabCut Image Segmentation)
    计算机视觉算法中的GrabCut图像分割(GrabCutImageSegmentation)引言图像分割是计算机视觉领域的一个重要任务,它的目标是将图像中的像素分成不同的区域或对象。GrabCut是一种经典的图像分割算法,它基于图割理论和高斯混合模型,能够有效地将图像中的前景和背景进行分离。本文将介绍Grab......
  • 平衡二叉树(Balanced Binary Tree)
    平衡二叉树(BalancedBinaryTree)平衡二叉树是一种特殊的二叉搜索树,它具有以下特点:每个节点的左子树和右子树的高度差不超过1。所有的子树也都是平衡二叉树。通过保持平衡性,平衡二叉树可以在最坏情况下仍然具有较好的性能,保证查找、插入和删除操作的时间复杂度为O(logn)。......
  • 二叉树 遍历 hdu-1710-Binary Tree Traversals
    给出二叉树的前序遍历和中序遍历,求后序遍历。。 算法:由前序遍历的第一个元素可确定左、右子树的根节点,参照中序遍历又可进一步确定子树的左、右子树元素。如此递归地参照两个遍历序列,最终构造出二叉树。 由前序和中序结果求后序遍历结果树的遍历: 给你一棵树的先......