首页 > 其他分享 >复杂数据结构

复杂数据结构

时间:2024-02-14 17:44:56浏览次数:31  
标签:log 复杂 序列 区间 操作 维护 数据结构 题意

复杂数据结构

一些巨大的数据结构题目

CF1336F Journey

题意:给定一棵树和 \(m\) 条链,求多少对链的交中包含的边 \(\geqslant k\)。

思路:首先对链交的情况进行分类。

第一种是 \(lca(x_1,y_1)\ne lca(x_2,y_2)\),我们在深度较大的 \(lca\) 处统计答案,那么我们把一条链的贡献记在端点向 \(lca\) 跳 \(k\) 步的位置,将其子树加,统计答案时求两端的答案即可。

第二种是 \(lca(x_1,y_1)=lca(x_2,y_2)\) 且 \(dfn[x_1]<dfn[x_2]<dfn[y_1]<dfn[y_2]\),那么就枚举 \(lca(x_1,x_2)=z\),从 \(z\) 向 \(y_2\) 走 \(k\) 步,再统计答案,那么具体做法就是对于所有 \(lca=x\) 的 \((u,v)\),建出 \(u\) 的虚树,把 \(v\) 挂在 \(u\) 上,同时维护 \(u\) 所在链,维护答案需要用线段树合并,维护链需要启发式合并。

第三种是 \(lca(x_1,y_1)=lca(x_2,y_2)\) 且 \(dfn[x_1]<dfn[y_1]<dfn[x_2]<dfn[y_2]\),那么和第一种类似,它们的交一定是从 lca 往下 \(k\) 个,在 \(y_1\) 处子树加,在 \(x_2\) 向上走 \(k\) 步统计答案即可。

P5385 [Cnoi2019] 须臾幻境

题意:求一段区间内的边形成的连通块数。

牛逼题。对于求联通块数量,有一点小trick。

首先,如果在一棵树上断掉一些边,那么连通块是就是点数减边数。而转到图上,可以任取一颗生成树森林,此时的点数减去在生成树森林上的边数就是连通块数。

回到这道题,我们需要的是求出一段时间内存在于生成树森林上的边数,我们可以先顺序加边,用LCT实时维护生成树森林,即如果当前要加入的边的两端已经联通,就删掉路径上最早加入的边,用主席树存下每一条边存在的时间就可以了。

P4848 崂山白花蛇草水

题意:在线带插入矩形第 \(k\) 大。

思路:在线带插入矩形第k大,看起来就不可做,结果可以直接暴力,搞两个数组然后插入到小的数组里,如果大于\(\sqrt{n}\)就归并到大数组里,询问时顺次查就可以了。

P5356 [Ynoi2017] 由乃打扑克

题意:区间加,区间 kth。

思路:算分块维护kth的经典trick:维护每一块排好序后的数组,查询时在每一块上二分。可以做到 \(O(n\sqrt{n}\log n)\)。

P7963 [NOIP2021] 棋局

思路:终于来写棋局了。

其实思路不难,就对于每类边分别维护一下。

对于第一类边,可以直接维护;

对于第二类边,可以用并查集,然后维护一下段头、段尾。

对于第三类边,用线段树合并即可。

关于去重,第一类点的去重比较容易,第二类点的去重稍微有点麻烦,不过因为这一类点在横或纵坐边上是连续的,因此在线段树上是一个区间,这样也好处理。

然后就是吃子的问题。对于第一类边,比较容易。对于第二类边,会被吃的最多只有 4 个,可以直接判。对于第三类边,我们在维护线段树的同时维护一下与当前连通块相邻的棋子集合,然后查询就是一个前缀查询。

P8868 [NOIP2022] 比赛

题意:给出两个序列 \(a,b\),每个区间的贡献是 \(a\) 序列中的最小值乘上 \(b\) 序列中的最大值,多次询问一个区间的所有子区间的权值和。

思路:以前以为是什么单调栈 + 线段树的神仙题,结果发现就是个用线段树暴力维护半群信息的题目。

先把询问离线下来,然后对 \(r\) 进行扫描线,维护当前所有 \(l\) 的答案。我们对 \(A,B\) 同时维护单调栈,这样对于两边都是一个区间覆盖,然后我们维护 \(S_{l,r}=\sum\limits_{r'=l}MaxA_{l,r'}\times MaxB_{l,r'}\),这样的话查询就是区间和。

考虑怎么动态维护这个信息。我们发现标记其实就是区间覆盖和区间 \(+=X\times Y\) 的复合,那么我们可以维护 \(sx,sy\) 表示 \(x,y\) 的区间覆盖标记, \(a_x,a_y,a_{x,y},a\) 分别表示区间和会增加多少,这样就差不多了。

P9247 [集训队互测 2018] 完美的队列

题意:有 \(n\) 个队列,每个队列有长度 \(a_i\),操作是往区间的每个队列中加入一个权值,求每个时刻队列中的权值种数。

思路:分块好题。

首先,可以转化成对于每一次 \(\text{push}\) 的操作,求出其最晚的被 \(\text{pop}\) 的时间。然后我们对序列分块。对于整块,我们用 \(\text{two\_pointers}\) 求出第 \(i\) 次操作后最早的时刻 \(j\) 满足 \((i,j]\) 中的操作可以让每个位置插入次数超过 \(a\)。对于散块,可以扫描线 + 树状数组上二分。块长取 \(\sqrt n\over \log n\) 时最优,复杂度是 \(m\sqrt{n}\log n\)。

P8360 [SNOI2022] 军队

题意:有两个数组 \(a,b\),有给 \(a\) 区间赋值,给区间中所有 \(a_i=x\) 的位置在 \(b\) 序列中加 \(y\),和查询 \(b\) 区间的区间和。

思路:考虑分块。

对于整块,可以考虑维护森林。初始颜色相同的点有相同的父节点,然后区间赋值操作就是合并这些颜色。

合并时,如果这个颜色块内没有,就需要新建一个点代表这个颜色,然后再合并。

对于区间加,在树上打标记即可。

对于散块,可以暴力下放标记,然后暴力维护操作和重构森林。

复杂度 \(O(n+q\sqrt{n})\)。

「JOISC 2016 Day 3」回转寿司

题意:给出一个有 N 个点的环,环上各点有一个初始权值 \(a_i\)。

给出 Q 个询问,每次询问给出一个区间 \([l,r]\) 和一个值 A ,对于 A 的变动定义如下(r 可能会小于 l 因为是环形):

for (int i = l; i <= r; i++) if(a[i] > A) swap(a[i],A);

对于每个询问,回答遍历完区间 \([l,r]\) 后 A 的最终值。

思路:先考虑 \(l=1,r=n\) 的部分分。假如我们只进行一次操作,那么我们最后得到的 A 一定是 A 和序列最大值中较大的那个,如果 A 比序列的最大值要小那么 A 会加入序列中,最大值会出来。如果进行多次操作,我们发现在这个情况下我们并不需要知道序列最终是什么样,我们只用知道这个序列中元素组成的可重集,就可以求出每次操作后 A 会是什么,于是我们用堆来维护即可。

然后考虑拓展到一般的情况。我们此时的目的肯定是想办法尽量只关注序列的可重集,而题目的数据范围不大,时限却很大,不难想到分块。

我们将序列分块,每一块维护元素的可重集和每次会将对整个块进行操作的 A 有哪些。对于每次操作,整块可以直接把 A 加入可重集,然后把最大值弹出,这就是操作完的 A。

重点是散块如何处理。我们顺次考虑每个数,然后顺次考虑 A,如果当前的 \(a_i>A\),这样 \(a_i\) 会变成 A,A 会变成 \(a_i\),然后会继续找到下一个比 \(a_i\) 小的 A,继续进行下去。不难发现,\(a_i\) 最终会变成 A 中最小值和 \(a_i\) 中较小的那个,如果 \(a_i\) 比 A 中最小的数还大就会和这个数交换。

设块长为 B,那么我们一次修改整块的复杂度是 \(O(\dfrac{n}{B}\log n)\),散块的复杂度是 \(O(B\log n)\),于是总复杂度就是 \(O(Q\sqrt{n}\log n+n\log n)\)。

P6109 [Ynoi2009] rprmq1

题意:有一个 \(n \times n\) 的矩阵 \(a\),初始全是 \(0\),有 \(m\) 次修改操作和 \(q\) 次查询操作,先进行所有修改操作,然后进行所有查询操作。

一次修改操作会给出 \(l_1,l_2,r_1,r_2,x\),代表把所有满足 \(l_1 \le i \le r_1\) 且 \(l_2 \le j \le r_2\) 的 \(a_{i,j}\) 元素加上一个值 \(x\)。

一次查询操作会给出 \(l_1,l_2,r_1,r_2\),代表查询所有满足 \(l_1 \le i \le r_1\) 且 \(l_2 \le j \le r_2\) 的 \(a_{i,j}\) 元素的最大值。

思路:因为是先加再询问,可以想到离线处理。那么可以想到枚举查询的左端点,然后扫描线,把加操作差分一下,那么就变成了求区间历史最值,这样做是 \(O((n^2+q)\log n)\)。

我们来分析一下这样做的实质。我们做的是加入 \([1,l)\) 的操作,不维护历史最值,然后加入 \([l,r]\) 的操作,维护历史最值,于是可以考虑用分治优化。

这里我们用猫树分治,因为这样只用把一个区间拆成 2 个而不是 log 个。处理区间 \([l,r]\) 时,我们先加入 \([l,mid]\) 的操作,然后打清空历史最值的标记,然后处理 \([r,mid]\) 的操作和询问,处理完后回退到打标记的状态,递归处理右子树,然后再处理 \([l,mid]\) 的操作和询问,最后递归左子树。

复杂度 \(O(n\log^2n+q\log n)\)。

标签:log,复杂,序列,区间,操作,维护,数据结构,题意
From: https://www.cnblogs.com/Xttttr/p/18015349

相关文章

  • 小清新数据结构
    小清新数据结构很小清新的数据结构题,主要是线段树和树状数组。CF840DDestiny题意:求区间内是否存在出现次数严格大于\(\dfrac{r-l+1}{k}\)的数。来自Alex_Wei老师的神仙思路:设\(d\)为严格大于\(\dfrac{r-l+1}{k}\)的最小数,那么如果一个数\(x\)至少出现了\(d\)次,就得满足一定......
  • 基础数据结构笔记
    链表与邻接表:树与图的存储 单链表  画个图就很好理解了   例题826.单链表acwing——826.单链表_awcing826-CSDN博客实现一个单链表,链表初始为空,支持三种操作:(1)向链表头插入一个数;(2)删除第k个插入的数后面的数;(3)在第k个插入的数后插入一个数现在要......
  • [数据结构] 串与KMP算法详解
    写在前面今天是农历大年初三,祝大家新年快乐!尽管新旧交替只是一个瞬间,在大家互祝新年快乐的瞬间,在时钟倒计时数到零的瞬间,在烟花在黑色幕布绽放的瞬间,在心底默默许下愿望的瞬间……跨入新的一年,并不意味了一切都会朝着更美好,也没有什么会从天而降,我们赋予了它这份意义,让它自然裹......
  • 【数据结构】C语言实现栈的相关操作
    栈栈是一种遵循先入后出逻辑的线性数据结构,是只能在表的一端进行插入和删除运算的线性表进行插入和删除的一端的称为栈顶,另一端称为栈底栈的操作规则是后进先出或者是先进后出栈可以用数组或者链表实现,用数组实现的叫做顺序栈,用链表实现的叫做链栈顺序栈表示(数组)在数组上......
  • 数据结构套路大赏
    现在感觉没啥写的,先留着以后会填的(一、线段树:1、线段树维护哈希2、线段树套DSU(其实没啥用)二、平衡树: 三、树状数组:1、树状数组二分(倍增),常数小,而且好写四、分块: 五、杂项:1、留意“取模”类的修改操作,每次取模会至少减半,哪怕看似暴力的解法也很有可能不高于nlogn2、扫......
  • 【Java 并发】【十】【JUC数据结构】【十】PriorityBlockingQueue 原理
    1 前言这节我们继续看看另一个队列 PriorityBlockingQueue,优先级的哈。2 PriorityBlockingQueue介绍PriorityBlockingQueue是带优先级的无界阻塞队列,每次出队都返回优先级最高或者最低的元素。其内部是使用平衡二叉树堆实现的,所以直接遍历队列元素不保证有序。默认使......
  • 【Java 并发】【十】【JUC数据结构】【九】ConcurrentLinkedQueue 原理
    1 前言JDK中提供了一系列场景的并发安全队列。总的来说,按照实现方式的不同可分为阻塞队列和非阻塞队列,前者使用锁实现,而后者则使用CAS非阻塞算法实现。这节我们来看看 ConcurrentLinkedQueue。2 ConcurrentLinkedQueue介绍ConcurrentLinkedQueue是线程安全的无界非阻......
  • 数据结构——第1章 绪论
    目录1.1数据结构的研究内容1.2基本概念和术语1.2.1数据、··元素、··项和··对象1.2.2数据结构1.2.3数据类型和抽象数据类型1.3抽象数据类型的表示与实现1.4算法和算法分析1.4.1算法的定义与特性1.4.2算法的时间复杂度1.4.3算法的空间复杂度1.5小结1.1数据结构的......
  • 学习 Redis 基础数据结构,不讲虚的。
    学习Redis基础数据结构,不讲虚的。一个群友给我发消息,“该学的都学了,怎么就找不到心意的工作,太难了”。很多在近期找过工作的同学一定都知道了,背诵八股文已经不是找工作的绝对王牌。企业最终要的是可以创造价值,或者首先需要干活的人,所以实战很重要。今天这篇文章就是给大家分享......
  • [数据结构] 队列
    队列的基本概念队列(Queue),是一种操作受限的线性表,只允许在表的一端进行插入,而在表的另一端进行删除。向队列中插入元素称为入队,删除元素称为出队。其操作特性是先进先出队列的常见操作:函数名功能InitQueue(*Q)初始化队列,构造一个空队列QQueueEmpty(Q)判断队列空......