【学习笔记】线段树分裂

news2024/9/19 17:31:57

前言

有线段树合并就应该有线段树分裂。它是线段树合并的逆过程。具体的,你需要以权值线段树中第 k 小的数为分界线,把线段树分成两半。

算法流程

和线段树上二分类似。假设原来的线段树为 u,要分裂出线段树 v

  1. 记左子树的权值为 val。
  2. 如果 k>val,那么分界线在右子树,那么左子树归 u,递归右子树,此时 k=k-val。
  3. 如果 k==val,那么分界线正好就是mid,那么左子树归 u,右子树归 v。
  4. 如果 k<val,那么分界线在左子树,那么右子树归 v,递归左子树
  5. 计算 u,v 的权值 tr[v].val=tr[u].val-k; tr[u].val=k;

核心代码

int split(int u,int v,int st,int ed,int k)
{
	if(u==0) return 0;
	int mid=st+ed>>1;
	tr.push_back(seg());
	v=tr.size()-1;
	int val=tr[tr[u].ls].val;
	if(k>val)
		tr[v].rs=split(tr[u].rs,tr[v].rs,mid+1,ed,k-val);
	else
		swap(tr[u].rs,tr[v].rs);
	if(k<val)
		tr[v].ls=split(tr[u].ls,tr[v].ls,st,mid,k);
	tr[v].val=tr[u].val-k;
	tr[u].val=k;
	return v;
}

【模板】线段树分裂

在这里插入图片描述
在这里插入图片描述

题解

操作0

先把线段树分裂成 ( 1 , x − 1 ) , ( x , n ) (1,x-1),(x,n) (1,x1),(x,n),再把 ( x , n ) (x,n) (x,n) 线段树分裂成 ( x , y ) , ( y + 1 , n ) (x,y),(y+1,n) (x,y),(y+1,n),最后合并线段树 ( 1 , x − 1 ) , ( y + 1 , n ) (1,x-1),(y+1,n) (1,x1),(y+1,n)

操作1

线段树合并

操作2

单点加

操作3

区间查

操作4

线段树上二分

代码

#include<bits/stdc++.h>
#define int long long
using namespace std;
const int N=1e6+7,inf=1e18;
int n,m;
struct seg
{
	int ls,rs,val;
	seg():ls(0),rs(0),val(0)
	{
	}
	seg(int a,int b,int c):ls(a),rs(b),val(c)
	{
	}
};
vector<seg> tr(2);
void update(int u)
{
	tr[u].val=tr[tr[u].ls].val+tr[tr[u].rs].val;
}
void insert(int u,int st,int ed,int x,int t)
{
	if(st==ed)
	{
		tr[u].val+=t;
		return;
	}
	int mid=st+ed>>1;
	if(x<=mid)
	{
		if(!tr[u].ls)
		{
			tr.push_back(seg());
			tr[u].ls=tr.size()-1;
		}
		insert(tr[u].ls,st,mid,x,t);
	}
	else
	{
		if(!tr[u].rs)
		{
			tr.push_back(seg());
			tr[u].rs=tr.size()-1;
		}
		insert(tr[u].rs,mid+1,ed,x,t);
	}
	update(u);
}
int query(int u,int st,int ed,int l,int r)
{
	if(l<=st&&ed<=r)
	{
		return tr[u].val;
	}
	int mid=st+ed>>1,res=0;
	if(l<=mid)
	{
		if(tr[u].ls)
			res+=query(tr[u].ls,st,mid,l,r);
	}
	if(mid<r)
	{
		if(tr[u].rs)
			res+=query(tr[u].rs,mid+1,ed,l,r);
	}
	return res;
}
void merge(int u,int v,int st,int ed)
{
	if(st==ed)
	{
		tr[u].val+=tr[v].val;
		return;
	}
	int mid=st+ed>>1;
	if(tr[u].ls&&tr[v].ls)
		merge(tr[u].ls,tr[v].ls,st,mid);
	else if(tr[v].ls)
		tr[u].ls=tr[v].ls;
	if(tr[u].rs&&tr[v].rs)
		merge(tr[u].rs,tr[v].rs,mid+1,ed);
	else if(tr[v].rs)
		tr[u].rs=tr[v].rs;
	update(u);
}
int split(int u,int v,int st,int ed,int k)
{
	if(u==0) return 0;
	int mid=st+ed>>1;
	tr.push_back(seg());
	v=tr.size()-1;
	int val=tr[tr[u].ls].val;
	if(k>val)
		tr[v].rs=split(tr[u].rs,tr[v].rs,mid+1,ed,k-val);
	else
		swap(tr[u].rs,tr[v].rs);
	if(k<val)
		tr[v].ls=split(tr[u].ls,tr[v].ls,st,mid,k);
	tr[v].val=tr[u].val-k;
	tr[u].val=k;
	return v;
}
int find(int u,int st,int ed,int k)
{
	if(k>tr[u].val||st>ed||k==0)
		return -1;
	if(st==ed)
	{
		return st;
	}
	int mid=st+ed>>1;
	int val=tr[tr[u].ls].val;
	if(k>val)
	{
		return find(tr[u].rs,mid+1,ed,k-val);
	}
	else
	{
		return find(tr[u].ls,st,mid,k);
	}
}
vector<int> rt(1);
void O_o()
{
	cin>>n>>m;
	rt.push_back(1);
	for(int i=1; i<=n; i++)
	{
		int x;
		cin>>x;
		insert(rt[1],1,n,i,x);
	}
	while(m--)
	{
		int op,id;
		cin>>op>>id;
		if(op==0)
		{
			int x,y;
			cin>>x>>y;
			rt.push_back(0);
			int now=rt.size()-1;
			int v1=query(rt[id],1,n,1,x-1),v2=query(rt[id],1,n,x,y);
			rt[now]=split(rt[id],rt[now],1,n,v1);
			int t=split(rt[now],0,1,n,v2);
			merge(rt[id],t,1,n);
		}
		else if(op==1)
		{
			int t;
			cin>>t;
			merge(rt[id],rt[t],1,n);
		}
		else if(op==2)
		{
			int x,q;
			cin>>x>>q;
			insert(rt[id],1,n,q,x);
		}
		else if(op==3)
		{
			int l,r;
			cin>>l>>r;
			cout<<query(rt[id],1,n,l,r)<<"\n";
		}
		else if(op==4)
		{
			int k;
			cin>>k;
			cout<<find(rt[id],1,n,k)<<"\n";
		}
		else assert(0);
	}
}
signed main()
{
	ios::sync_with_stdio(false); cin.tie(0); cout.tie(0);
	cout<<fixed<<setprecision(12);
	int T=1;
//	cin>>T;
	while(T--)
	{
		O_o();
	}
}

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

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

相关文章

CodeMeter助力软件授权与IP保护,保障工业自动化与物联网安全

随着工业自动化的飞速发展&#xff0c;Hilscher的开放工业4.0联盟&#xff08;OI4&#xff09;旗舰店应运而生&#xff0c;将应用商店模式引入工业领域。凭借CodeMeter授权和加密技术的支持&#xff0c;该商店为工业用户提供了一个安全且开放的应用程序和解决方案平台。该平台不…

超声波清洗机哪个品牌好用又实惠?精选业内四款优质清洗机推荐

超声波清洗机作为一种创新的清洁解决方案&#xff0c;凭借其深入微观的清洁效能、简便的操作方式以及对物品的细腻呵护&#xff0c;正逐渐成为广受喜爱的清洁良品。不过&#xff0c;市面上品牌林立、型号多样&#xff0c;价格亦波动不一&#xff0c;这无疑为消费者选购时平添了…

ATT&CK靶机实战系列之vulnstack2

声明: 本文章只是用于网络安全交流与学习&#xff0c;若学者用学到的东西做一些与网络安全不相关的事情&#xff0c;结果均与本人无关&#xff01;&#xff01;&#xff01; 靶场环境: 使用kali作为hacker的攻击机器&#xff0c;来对web pc dc进行攻击。 这里声明一下: 关于…

Cesium billboard 自定义shader实现描边效果

Cesium billboard 自定义shader实现描边效果 uniform sampler2D u_atlas;uniform vec2 dimensions;in vec2 v_textureCoordinates;in vec4 v_pickColor;in vec4 v_color;in float v_splitDirection;void main(){if (v_splitDirection < 0.0 && gl_FragCoord.x > …

一文详解可视化大屏技术在地震监测中的作用!

昨天&#xff0c;安徽合肥市肥东县发生地震&#xff0c;震级达到4.7级&#xff0c;震源深度12千米。这一事件再次提醒我们&#xff0c;地震的威胁无处不在&#xff0c;及时有效的地震预警对于减少灾害损失至关重要。四川省作为地震活动频繁的地区&#xff0c;近年来在地震监测和…

配置环境-keil

配置keil -- 先将keil安装配置好&#xff0c;包括库 一、STM32 -- STM32是意法半导体&#xff08;意大利&#xff09;采用ARM公司设计的内核&#xff0c;设计一系列32位单片机芯片。 1、STM32开发的几种方式 2、STM32寄存器和库函数版本的工程创建 新建文件夹 复制相关文件…

【机器学习(八)】分类和回归任务-因子分解机(Factorization Machines,FM)-Sentosa_DSML社区版

文章目录 一、算法概念二、算法原理&#xff08;一&#xff09; FM表达式&#xff08;二&#xff09;时间复杂度&#xff08;三&#xff09;回归和分类 三、算法优缺点&#xff08;一&#xff09;优点&#xff08;二&#xff09;缺点 四、FM分类任务实现对比&#xff08;一&…

【2024华为杯研究生数学建模竞赛】比赛思路、代码、论文更新中.....

目录 赛中助攻华为杯常用建模算法&#x1f5d2;️&#x1f5d2;️历年优秀论文⭐⭐论文模板1&#xff09;论文模板2&#xff09;基础画图能力 绘图与数据分析软件SPSSPRO 2024研究生数学建模竞赛时间为9月21日&#xff08;周六&#xff09;8:00至9月25日&#xff08;周三&#…

互联网前端之 CSS 常见属性与三层结构

目录 现在互联网前端三层 CSS 常见属性 关注作者微信公众号&#xff0c;开启探索更多 CSS 知识的精彩之旅。在这里&#xff0c;你将收获丰富的CSS专业内容&#xff0c;深入了解这一网页开发语言的奥秘&#xff0c;不断拓展你的知识边界&#xff0c;提升技能水平。快来关注吧&…

对想学习人工智能或者大模型技术从业者的建议,零基础入门到精通,收藏这一篇就够了

“ 技术的价值在于应用&#xff0c;理论与实践相结合才能事半功倍**”** 写这个关于AI技术的公众号也有差不多五个月的时间了&#xff0c;最近一段时间基本上都在保持日更状态&#xff0c;而且写的大部分都是关于大模型技术理论和技术方面的东西。‍‍‍‍‍‍‍‍‍ 然后最近…

Windows安全日志分析(事件ID详解)

目录 如何查看Windows安全日志 常见事件ID列表 事件ID 1116 - 防病毒软件检测到恶意软件 事件ID 4624 - 账户登录成功 事件ID 4625 - 账户登录失败 事件ID 4672 - 为新登录分配特殊权限 事件ID 4688 - 新进程创建 事件ID 4689 - 进程终止 事件ID 4720 - 用户账户创建 …

3款免费的GPT类工具

前言 随着科技的飞速发展&#xff0c;人工智能&#xff08;AI&#xff09;的崛起与发展已经成为我们生活中不可或缺的一部分。它的出现彻底改变了我们与世界互动的方式&#xff0c;并为各行各业带来了前所未有的便利。 一、Kimi 网址&#xff1a;点我前往 国产AI模型Kimi是一…

C++第八节课 日期类的补充

在上节课我们提交的代码中&#xff0c;还有着一些不足&#xff1a; 如果我们想要运行下面的函数&#xff1a; void test4() {Date d1(2023, 5, 5);d1 -50;d1.Print();Date d2(2023, 5, 5);d2 - -50;d2.Print(); } 我们发现之前的代码没有考虑day为负数的情况&#xff0c;可以…

浅谈红外测温技术在变电站运维中的应用

0引言 随着市场经济的繁荣发展&#xff0c;社会对电力的需求持续增长。城市供电网络的规模和用电设备的总量也在不断扩大&#xff0c;这导致城市电力系统中潜在的网络安全隐患日益增多。作为电力系统核心组成部分的变压器&#xff0c;其安全、稳定的工作直接关系到电能的质量和…

俄罗斯的Alexey V. Gubin开发的数据恢复文件-零假设恢复只读模式下对扫描/恢复数据起作用-供大家学习研究参考

// 主要特征 // Windows FAT,NTFS,Linux ext2 / 3/4和XFS卷格式化的驱动器或“ RAW文件系统”外部驱动器,USB拇指驱动器和存储卡带有ZAR Data Recovery免费版本的数码照片恢复RAID数据恢复NAS数据恢复MBR损坏数据恢复具有多个逻辑驱动器的分区表恢复支持长文件名和国家文件名…

图神经网络模型应用(8)--2

1.自然语言处理相关的应用 文本分类是自然语言处理中的一个经典应用&#xff0c;图神经网络常用的标准数据集里就包含引用网络中论文的分类&#xff0c;但是作为机器学习领域的通用模型测试数据集&#xff0c;它们并没有充分利用文本本身的结构(每个文档只是用词袋特征来表示),…

【微处理器系统原理和应用设计第十五讲】模拟/数字转换器

一、基础概念 1、概念 ADC将真实世界的例如温度、压力、声音或者图像等的模拟信号转换为更容易储存、处理和发射的数字形式。 2、分类 间接ADC&#xff1a;先将输入模拟电压转化为时间/频率等数字信号&#xff0c;例如双积分型ADC 直接ADC&#xff1a;直接转化为数字量&am…

Pandas的读写数据

目录 读写文件的类型 Excel写 API 准备数据 1.直接写入(默认有索引和标题) 2.写入(去掉索引) 3.写入(去掉索引和标题) 4.写入(去掉索引和标题,指定表单信息) Excel读 API 1.读(默认带有索引和标题) 2.读(指定索引项) 3.读(碰到无标题列和无索引列,指定索引列,标题列…

C++速通LeetCode中等第7题-和为K的子数组(巧用前缀和)

巧用哈希表与前缀和&#xff0c;前缀和差为k的两个序号之间的数组就是满足条件的子数组&#xff0c;用哈希表来存放每个序号的前缀和。 前缀和就是头元素到当前序号子数组元素的和 class Solution { public:int subarraySum(vector<int>& nums, int k) {unordered_…

【软件测试】测试的岗位有哪些?

求职入口有很多&#xff1a;相关企业官网、求职软件、校招、公众号等等。 下面就在某招聘网站上看看测试有哪些岗位吧&#xff01; 测试只是一个统称&#xff0c;在测试下面还有很多细分岗位。 但是测试的岗位主要分为以下俩个方面&#xff1a; 软件测试开发工程师&#xff…