左偏树与可持久化左偏树

上次thucamp有一道题:有n+1个multiset,编号从0开始,一开始都为空。第i次操作(i=1,2,…,n)有三种可能(输入确定),令 s i = s j ⋃ x s_i=s_{j} \bigcup {x} si=sjx, 或者 s i = s x ⋃ s y s_i=s_x \bigcup s_y si=sxsy,或者 s i = s j s_i=s_j si=sj然后删除最小元素,每次操作都要查询一次最小值。

因为当时不会左偏树,所以我尝试用线段树来写,但是发现线段树并不适合这一道题。比如说,比如某两个multiset都有1~10000的每个数,那么线段树上就会形成一个有接近10000个节点的满二叉树,然后每次合并都要把全部的点遍历一次,这样必然会超时。毕竟,题目只需要查询最小值,而线段树能够满足更多的功能。

后面听说要用左偏树来做,于是我就打算上网学习一下。

我先做了洛谷的一个模板,然后就意识到这个东西完全可以用于可持久化。

P3377 【模板】左偏树/可并堆

其实这道题有很多的方法可以做。

但是学习完左偏树以后,感觉这种算法非常符合直觉,感觉顺利成章的想就能想出来,而且代码很短,十分优美。

左偏树也叫可并堆,顾名思义,它就是一种堆,但是合并非常的快。那么我们现在先思考一下怎么合并两个堆。一种常见的想法就是用启发式合并,这样的时间复杂度是双log的,而且不能可持久化。

现在考虑直接合并两个堆,可以利用堆的高度是log的性质合并。而不是一个个节点的插进去。假设我们要求的是小根堆,那么我们肯定要先固定小的那个点,然后把根值更大的一堆和它其中的一个儿子合并,那么我们应该如何选择儿子呢?特别地,如果存在空儿子,那么我们就直接把大的根作为空儿子就行。也就是说,只要我们合并能遇到空儿子,就能结束合并,因此我们要找到最近的空儿子。很显然而且很重要的,假设这个最近的距离为d,那么堆至少有 2 d − 1 2^d-1 2d1个点,这就说明 d d d的大小就是log的。

综上,我们可以维护距离每个点最近的空儿子,合并的时候就可以往那边走。左偏树的做法就是依靠对换左右儿子,始终把这个最近的空儿子放在右边,因此在左偏树中,每次合并都只需要在确定哪个为当前根节点以后往右走即可,时间复杂度是单log的。

现在说一下几种操作的具体实现:
1.合并两个堆,这就是我们前面说的。
2.删除根节点,就是合并左右儿子。
3.查找每个节点所在的根节点。因为深度不确定,所以不能在左偏树上面直接找,而是要专门打一个并查集维护。
4.删除某个节点。(我自己的想法)合并其左右儿子,把节点从堆中删除,把堆的根节点和已经合并的左右儿子再次合并(虽然常数大但是很好记,而且不用专门打一个函数)。
5.想查询历史版本。我们的目的是不改变原有节点的关系,因此对于任意要修改的节点,新建一个节点来代替它。

洛谷P3377

#include<bits/stdc++.h>
#define rep(i,x,y) for(int i=x;i<=y;i++)
#define dwn(i,x,y) for(int i=x;i>=y;i--)
#define ll long long
using namespace std;
template<typename T>inline void qr(T &x){x=0;int f=0;char s=getchar();while(!isdigit(s))f|=s=='-',s=getchar();while(isdigit(s))x=x*10+s-48,s=getchar();x=f?-x:x;
}
int cc=0,buf[31];
template<typename T>inline void qw(T x){if(x<0)putchar('-'),x=-x;do{buf[++cc]=int(x%10);x/=10;}while(x);while(cc)putchar(buf[cc--]+'0');
}
const int N=1e5+10;
int n,m;
struct heap{int lc,rc,d,val;
}t[N];
int fa[N],rt[N];
bool v[N];
int findfa(int x){return x==fa[x]?x:fa[x]=findfa(fa[x]);}
int merge(int x,int y){if(!x||!y)return x|y;if(t[x].val>t[y].val)swap(x,y);t[x].rc=merge(t[x].rc,y);if(t[t[x].lc].d<t[t[x].rc].d)swap(t[x].lc,t[x].rc);t[x].d=t[t[x].rc].d+1;return x;
}
void solve(){qr(n),qr(m);rep(i,1,n){int x;qr(x);t[i].lc=t[i].rc=t[i].d=0;t[i].val=x;fa[i]=i;rt[i]=i;}while(m--){int op;qr(op);if(op==1){int x,y;qr(x),qr(y);if(v[x]||v[y]||findfa(x)==findfa(y))continue;x=rt[findfa(x)],y=rt[findfa(y)];rt[findfa(x)]=merge(x,y);fa[findfa(y)]=findfa(x);}else{int x;qr(x);if(v[x]){puts("-1");continue;}x=rt[findfa(x)];rt[findfa(x)]=merge(t[x].lc,t[x].rc);v[x]=1;qw(t[x].val);puts("");}}
}
int main(){int tt;tt=1;while(tt--)solve();return 0;
}

thucampd2G

#include<bits/stdc++.h>
#define rep(i,x,y) for(int i=x;i<=y;i++)
#define dwn(i,x,y) for(int i=x;i>=y;i--)
#define ll long long
using namespace std;
template<typename T>inline void qr(T &x){x=0;int f=0;char s=getchar();while(!isdigit(s))f|=s=='-',s=getchar();while(isdigit(s))x=x*10+s-48,s=getchar();x=f?-x:x;
}
int cc=0,buf[31];
template<typename T>inline void qw(T x){if(x<0)putchar('-'),x=-x;do{buf[++cc]=int(x%10);x/=10;}while(x);while(cc)putchar(buf[cc--]+'0');
}
const int N=2e5+10;
struct heap{int lc,rc,d,val;
}t[N*70];int cnt,rt[N];
int n;
int merge(int x,int y){if(!x||!y)return x|y;if(t[x].val>t[y].val)swap(x,y);int p=++cnt;t[p]=t[x];t[p].rc=merge(t[p].rc,y);if(t[t[p].lc].d<t[t[p].rc].d)swap(t[p].lc,t[p].rc);t[p].d=t[t[p].rc].d+1;return p;
}
void print(int x){if(!x)return;cout<<t[x].val<<" ";print(t[x].lc);print(t[x].rc);
}
void solve(){qr(n);int s=0,ans=0;rep(i,1,n){int op;qr(op);if(op==1){int a,b;qr(a),qr(b);int pre,num;//qr(pre),qr(num);pre=(a+s)%i;num=(b+17*s)%(1000000001);t[++cnt].val=num;rt[i]=merge(rt[pre],cnt);ans=t[rt[i]].val;qw(ans);puts("");}else if(op==2){int a,b;qr(a),qr(b);int x,y;//qr(x),qr(y);x=(a+s)%i;y=(b+13*s)%i;rt[i]=merge(rt[x],rt[y]);if(!rt[i]){puts("empty");ans=0;}else{ans=t[rt[i]].val;qw(ans);puts("");}}else{int a;qr(a);int pre=(a+s)%i;// int pre;qr(pre);if(!rt[pre]){puts("empty");ans=0;}else{rt[i]=merge(t[rt[pre]].lc,t[rt[pre]].rc);ans=t[rt[pre]].val;qw(ans);puts("");}}s=(s+ans)%239017;}
}
int main(){// freopen("in.in","r",stdin);int tt;tt=1;while(tt--)solve();return 0;
}

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

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

相关文章

erlang学习:Linux命令学习4

顺序控制语句学习 if&#xff0c;else对文件操作 判断一个文件夹是否存在&#xff0c;如果存在则进行删除&#xff0c;如果不存在则创建该文件夹&#xff0c;并复制一份该脚本后&#xff0c;删除该脚本 if [ -d "/erlangtest/testdir"]; then echo "删除文件夹…

【路径规划】绘制算术和几何布朗运动- 绘制布朗桥、2D 和 3D 布朗运动- 绘制一些随机路径

摘要 本文演示了如何生成和绘制布朗运动、几何布朗运动和布朗桥的随机路径。这些随机路径广泛应用于金融、物理和工程领域&#xff0c;用于模拟随机过程。实验结果包括了多条随机路径的示例&#xff0c;展示了不同类型的布朗运动的特征。 理论 1. 布朗运动 (Brownian Motion…

构建高效房屋租赁系统:Spring Boot应用

1 绪论 1.1 研究背景 中国的科技的不断进步&#xff0c;计算机发展也慢慢的越来越成熟&#xff0c;人们对计算机也是越来越更加的依赖&#xff0c;科研、教育慢慢用于计算机进行管理。从第一台计算机的产生&#xff0c;到现在计算机已经发展到我们无法想象。给我们的生活改变很…

如何在NXP源码基础上适配ELF 1开发板的UART功能

UART即通用异步收发器&#xff0c;是一种支持全双工串行通信协议的接口。在i.MX6ULL处理器平台上&#xff0c;该处理器原生支持多达8路的UART接口&#xff0c;提供了丰富的串行通信能力。 针对ELF 1开发板&#xff0c;实际引出了4路UART接口供开发者使用&#xff0c;具体包括U…

Node-RED-L2-Node-RED在Linux系统启动时自动运行

Node-RED在Linux系统启动时自动运行 目的步骤1创建服务文件&#xff1a;2重新加载服务&#xff1a;3启用服务&#xff1a;4启动Node-RED服务&#xff1a;5检查服务状态&#xff1a;6其他说明7如果没启动正确的Node-RED执行路径&#xff1a;确保使用绝对路径&#xff1a; 检查用…

Flutter 约束布局

配置插件依赖 设置组件大小 通过属性 childConstraints 实现 分别设置 约束布局一 和 约束布局二 大大小为:160 和 200 点击查看代码文件 class SummaryPageState extends State<SummaryPage1> {ConstraintId constraintId_1 = ConstraintId(ConstraintId_1);Constrain…

易航网址导航系统V2.45完美去授权版

简介 易航网址导航系统V2.45完美去授权版 界面

Spring(看这一篇就够了)

Spring 概述 Spring 是最受欢迎的企业级 Java 应用程序开发框架&#xff0c;数以百万的来自世界各地的开发人员使用 Spring 框架来创建性能好、易于测试、可重用的代码。 Spring 框架是一个开源的 Java 平台&#xff0c;它最初是由 Rod Johnson 编写的&#xff0c;并且…

SFUD库移植

1.源码 GitHub - armink/SFUD: An using JEDECs SFDP standard serial (SPI) flash universal driver library | 一款使用 JEDEC SFDP 标准的串行 (SPI) Flash 通用驱动库 2.介绍 这个通用驱动库,实际就是帮你封装好了读写spiflash的函数, 我们只需要对接以下底层,就可以轻松…

【个人笔记】线程和线程池的状态以及转换方式

线程和线程池的状态是不一样的&#xff01;&#xff01; 线程有 6 种状态&#xff0c;查看Thread的State枚举类&#xff1a; NEW&#xff1a;创建后没启动的线程就处于这种状态RUNNABLE&#xff1a;正在java虚拟机中执行的线程就处于这种状态BLOCKED&#xff1a;受阻塞并等待…

Observability:构建下一代托管接入服务

作者&#xff1a;来自 Elastic Vishal Raj, Marc Lopez Rubio 随着无服务器&#xff08;serverless&#xff09;的引入&#xff0c;向 Elastic Cloud 发送可观察性数据变得越来越容易。你可以在 Elastic Cloud Serverless 中创建一个可观察性无服务器项目&#xff0c;并将可观察…

【Java】虚拟机(JVM)内存模型全解析

目录 一、运行时数据区域划分 版本的差异&#xff1a; 二、程序计数器 程序计数器主要作用 三、Java虚拟机 1. 虚拟机运行原理 2. 活动栈被弹出的方式 3. 虚拟机栈可能产生的错误 4. 虚拟机栈的大小 四、本地方法栈 五、堆 1. 堆区的组成&#xff1a;新生代老生代 …

Ubuntu磁盘不足扩容

1.问题 Ubuntu磁盘不足扩容 2.解决方法 安装一下 sudo apt-get install gpartedsudo gparted

Mysql梳理6——order by排序

目录 6 order by排序 6.1 排序数据 6.2 单列排序 6.3 多行排列 6 order by排序 6.1 排序数据 使用ORDER BY字句排序 ASC&#xff08;ascend&#xff09;:升序DESC(descend):降序 ORDER BY子句在SELECT语句的结尾 6.2 单列排序 如果没有使用排序操作&#xff0c;默认…

C语言课程设计题目一:职工信息管理系统设计

文章目录 题目一&#xff1a;职工信息管理系统设计代码块employeeManagement.hemployeeManage.ctest.c 调试验证录入信息&#xff0c;并浏览验证职工号唯一保存职工信息&#xff0c;加载职工信息按职工号进行查询根据id删除职工修改职工信息 题目一&#xff1a;职工信息管理系统…

下水道内缺陷识别检测数据集 yolo数据集 共2300张

下水道内缺陷识别检测数据集 yolo数据集 共2300张 下水道内部缺陷识别数据集&#xff08;Sewer Interior Defect Recognition Dataset, SIDRD&#xff09; 摘要 SIDRD 是一个专门针对下水道内部缺陷识别的数据集&#xff0c;旨在为城市基础设施维护和管理提供一个标准化的训练…

VmWare安装虚拟机保姆级教程(centos7,虚拟机网络设置,虚拟机桌面显示)

VMWare下载&#xff1a; 下载 VMware Workstation Pro - VMware Customer Connect 安装包&#xff1a;&#xff08;16的版本&#xff09;免费&#xff01;&#xff08;一个赞就行&#xff09; 一直点下一步即可&#xff0c;注意修改一下安装位置就好 二、安装虚拟机 安装虚…

论文复现:考虑电网交互的风电、光伏与电池互补调度运行(MATLAB-Yalmip-Cplex全代码)

论文复现:考虑电网交互的风电、光伏与电池储能互补调度运行(MATLAB-Yalmip-Cplex全代码) 针对风电、光伏与电化学储能电站互补运行的问题,已有大量通过启发式算法寻优的案例,但工程上更注重实用性和普适性。Yalmip工具箱则是一种基于MATLAB平台的优化软件工具箱,被广泛应用…

[uni-app]小兔鲜-02项目首页

轮播图 轮播图组件需要在首页和分类页使用, 封装成通用组件 准备轮播图组件 <script setup lang"ts"> import type { BannerItem } from /types/home import { ref } from vue // 父组件的数据 defineProps<{list: BannerItem[] }>()// 高亮下标 const…

【React】Ant Design 5.x版本drawer抽屉黑边问题

环境 antd: ^5.14.1react: ^18 问题情况 <Drawer open{open} closable{false} mask{false} width{680}getContainer{props.getContainer || undefined}><p>Some contents...</p><p>Some contents...</p><p>Some contents...</p> …