左偏树与可持久化左偏树

上次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.xdnf.cn/news/1541968.html

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

(C++17) optional 的 3 种用法

文章目录 *️⃣前言*️⃣3 种主流用法1️⃣函数返回值2️⃣函数参数3️⃣类成员 ⭐END&#x1f31f;跋&#x1f31f;交流方式 *️⃣前言 在 C17 中标准化了 std::optional。该类型可以容纳一种类型&#xff0c;且判断是否有无。 若使用的标准在低于 C17 则可以使用 Abseil 的…

浅谈递推法

递推法 递推法是一种数学方法&#xff0c;用于通过利用已知的初始条件和递推关系来计算要求中的每一项。以数列来举例&#xff0c;在递推法中&#xff0c;它的思想很简单&#xff1a;我们首先知道数列的第一项&#xff08;初始条件&#xff09;&#xff0c;然后通过一个规律&a…

GEE 数据集:人类造成的热带潮湿森林退化程度的估计

目录 简介 摘要 代码 结论 数据和代码 引用 网址推荐 0代码在线构建地图应用 机器学习 人类造成的热带潮湿森林退化程度超出了先前的估计 简介 选择性采伐、火灾和边缘效应造成的热带森林退化是碳和生物多样性损失的主要驱动因素1,2,3,其年增长率可与森林砍伐相媲美…

Golang | Leetcode Golang题解之第424题替换后的最长重复字符

题目&#xff1a; 题解&#xff1a; func characterReplacement(s string, k int) int {cnt : [26]int{}maxCnt, left : 0, 0for right, ch : range s {cnt[ch-A]maxCnt max(maxCnt, cnt[ch-A])if right-left1-maxCnt > k {cnt[s[left]-A]--left}}return len(s) - left }f…

【算法题】63. 不同路径 II-力扣(LeetCode)-”如果起点有障碍物,那么便到不了终点“

【算法题】63. 不同路径 II-力扣(LeetCode)-”如果起点有障碍物&#xff0c;那么便到不了终点“ 1.题目 下方是力扣官方题目的地址 63. 不同路径 II 一个机器人位于一个 m x n 网格的左上角 &#xff08;起始点在下图中标记为 “Start” &#xff09;。 机器人每次只能向下…

【全网最全】2024年华为杯研赛A题成品论文获取入口(后续会更新)

您的点赞收藏是我继续更新的最大动力&#xff01; 一定要点击如下的卡片&#xff0c;那是获取资料的入口&#xff01; 点击链接加入【2024华为杯研赛资料汇总】&#xff1a;https://qm.qq.com/q/hMgWngXvcQhttps://qm.qq.com/q/hMgWngXvcQ你是否在寻找数学建模比赛的突破点&am…

BUUCTF逆向wp [WUSTCTF2020]Cr0ssfun

第一步 查壳&#xff0c;本题是64位&#xff0c;无壳。 第二步 查看主函数&#xff0c;点开看主函数&#xff0c;没什么东西。 左边表里面看到好几个i开头的函数&#xff08;红色方框里面&#xff09;&#xff0c;点开看后每个函数的最后末尾&#xff08;图中红色椭圆圈那里&a…

(笔记自用)位运算总结+LeetCode例题:颠倒二进制位+位1的个数

一.位运算总结: 在解题之前理解一下为什么需要位运算&#xff1f;它的本质是什么&#xff1f; 力扣上不少位运算相关的题&#xff0c;并且很多题也会用到位运算的技巧。这又是为什么&#xff1f; 位运算的由来 在计算机里面&#xff0c;任何数据最终都是用数字来表示的&…

在Java中基于GeoTools的Shapefile读取乱码的问题解决办法

目录 前言 1、Shapefile属性字段编码的情况&#xff1a; 一、Shp文件常见的字符集编码 1、System编码 2、ISO-8859-1编码 3、UTF-8编码 二、GeoTools解析实战 1、未进行字符处理 2、乱码问题的解决 3、转码支持 4、属性字段编码结果 三、总结 前言 文件编码&#x…

分布式锁优化之 使用lua脚本改造分布式锁保证判断和删除的原子性(优化之LUA脚本保证删除的原子性)

文章目录 1、lua脚本入门1.1、变量&#xff1a;弱类型1.2、流程控制1.3、在lua中执行redis指令1.4、实战&#xff1a;先判断是否自己的锁&#xff0c;如果是才能删除 2、AlbumInfoApiController --》testLock()3、AlbumInfoServiceImpl --》testLock() 1、lua脚本入门 Lua 教程…

Linux基础命令以及常识

镜像站点服务器&#xff08;相当于下载的网址&#xff09;也可叫软件源 vim /etc/apt/sources.list 索引文件(网络服务器在本地的缓存) 服务器软件源在本地列出来一个清单&#xff0c;以便于主机进行查询操作 cd /var/lib/apt/lists/ 下载软件包默认存放路径 cd /var/cache/a…

认识NDK

什么是NDK&#xff08;Native Development Kit&#xff09; The Android NDK is a toolset that lets you implement parts of your app in native code, using languages such as C and C. &emdp; Android NDK 是一个工具集&#xff0c;可让您使用 C 和 C 等语言以原生代…

重型工程车辆数据集

重型工程车辆数据集&#xff0c;内含Bull_dozer&#xff08;推土机&#xff09;, Dumb_truck&#xff08;卡车&#xff09;, Excavator&#xff08;挖掘机&#xff09;, Grader&#xff08;平地机&#xff09;, Loader&#xff08;转载机&#xff09;, Mobile_crane&#xff08…

『功能项目』QFrameWork拾取道具UGUI【69】

本章项目成果展示 我们打开上一篇68QFrameWork扔到地上UGUI的项目&#xff0c; 本章要做的事情是实现当物品在地上时&#xff0c;点击物品将对应物品转移到道具栏中 制作一个提示UI界面 添加Button组件设置为点击即将父物体隐藏 拖拽到文件夹中在场景中删除 创建脚本&#xf…

架构师:使用 Zookeeper 实现分布式锁的技术指南

1、简述 在分布式系统中,多个节点可能需要访问共享资源或执行需要互斥的操作,为了避免竞争导致数据不一致或资源争用,我们需要一种机制来协调各个节点对资源的访问。分布式锁是用于解决这种竞争问题的关键技术,它确保在同一时间只有一个节点能够访问或修改共享资源。 2、Z…

Ansible部署与应用基础

由于互联网的快速发展导致产品更新换代速度逐步增长&#xff0c;运维人员每天都要进行大量的维护操作&#xff0c;按照传统方式进行维护使得工作效率低下。这时部署自动化运维就 可以尽可能安全、高效的完成这些工作。 一、Ansible概述 1.什么是Ansible Ansible 是基于 Pytho…

Matplotlib绘图基础

1、散点图 绘制散点图是数据可视化中非常常见的操作&#xff0c;它用于显示两组数据之间的关系。Matplotlib 提供了 plt.scatter() 函数&#xff0c;可以轻松绘制散点图。以下是一个基础的散点图示例代码&#xff0c;并包含了一些优化可视化呈现的技巧。 import matplotlib.p…

Python 如何调用讯飞星火大模型API

1 讯飞星火简介 讯飞星火是科大讯飞推出的一款先进的人工智能大模型&#xff0c;它具备强大的语言理解和知识问答能力&#xff0c;能够在多种场景中提供智能化服务。2024年6月27日&#xff0c;科大讯飞发布了讯飞星火大模型V4.0版本&#xff0c;全面对标GPT-4 Turbo。现有的模…

某采招网爬虫数据采集逆向

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、pandas是什么&#xff1f;二、使用步骤 1.引入库2.读入数据总结 前言 目标网站 aHR0cHM6Ly9zZWFyY2guYmlkY2VudGVyLmNvbS5jbi9zZWFyY2g/a2V5d29yZHM9JWU0…

医院伤员消费点餐限制———未来之窗行业应用跨平台架构

一、点餐上限 医院点餐上限具有以下几方面的意义&#xff1a; 1. 控制成本 - 有助于医院合理规划餐饮预算&#xff0c;避免食物的过度供应造成浪费&#xff0c;从而降低餐饮成本。 2. 保障饮食均衡 - 防止患者或陪护人员过度点餐某一类食物&#xff0c;有利于引导合…