树状数组专题

折叠

在这里插入图片描述
区间修改,区间查询,这一类题通常都可以使用线段树解决,但对于此题,树状数组同样可以,而且常数较小,代码简单。
思路:
考虑使用树状数组去维护差分数组,即对于 a i a_i ai,我们使用树状数组去维护 ∣ a i − a i − 1 ∣ |a_i-a_{i-1}| aiai1的值。
对于修改,我们对一段区间进行修改的时候,能对结果产生影响的只有左右端点,因为绝对值之差相互抵消了。
所以我们考虑修改时,端点的影响即可。
对于 a l a_l al,若 a l − 1 ≤ a l a_{l-1}\leq a_l al1al,那么我们在对 a l a_l al进行加一后,我们会发现, ∣ a l − a l − 1 ∣ |a_l-a_{l-1}| alal1的值同样会加一,所以我们正常给 a l a_l al加一即可。
反之, a l − 1 > a l a_{l-1}>a_{l} al1>al,那么在给 a l a_l al加一的时候, ∣ a l − a l − 1 ∣ |a_l-a_{l-1}| alal1的值会减小,所以此时我们需要给该值减一。
对于 r , r + 1 r,r+1 r,r+1位置的分析同理。
同时,又因为查询需要输出 a l a_l al的值,所以我们再开一个树状数组去单独维护每个值的大小即可。

#include <bits/stdc++.h>

using namespace std;
const int N = 2e5 + 5;
typedef long long ll;
typedef pair<ll, ll> pll;
typedef array<ll, 3> p3;
int mod = 1e9+7;
const int maxv = 4e6 + 5;
// #define endl "\n"

struct MIT
{
ll tr[N];
int lowbit(int x) {
    return x & (-x);
}

void add(int u, int v) {
    for (int i = u; i < N; i += lowbit(i)) {
        tr[i] += v;
    }
}

ll query(int x) {
    ll res = 0;

    for (int i = x; i > 0; i -= lowbit(i)) {
        res += tr[i];
    }

    return res;
}
};

MIT t1,t2;

void solve()
{
	int n,q;
	cin>>n>>q;
	vector<int> a(n+5);
	for(int i=1;i<=n;i++){
		cin>>a[i];
		t1.add(i,a[i]-a[i-1]);
		t2.add(i,abs(a[i]-a[i-1]));
	}
	while(q--){
		int op,l,r;
		cin>>op>>l>>r;
		if(op==1){
			cout<<t1.query(l)+t2.query(r)-t2.query(l)<<endl;
		}
		else{
			ll c1=t1.query(l-1),c2=t1.query(l);
			if(c2-c1>=0) t2.add(l,1);
			else t2.add(l,-1);
			c1=t1.query(r),c2=t1.query(r+1);
			if(c2-c1>0) t2.add(r+1,-1);
			else t2.add(r+1,1);
			t1.add(l,1),t1.add(r+1,-1);
		}
	}
}

int main()
{
    ios::sync_with_stdio(0);
	cin.tie(0);
	cout.tie(0);
	int t;
	t=1;
	//cin>>t;

	while(t--){
		solve();
	}
   	system("pause");
    return 0;
}

Mex and Update

问题陈述

给你一个长度为 N N N 的序列 A = ( A 1 , A 2 , … , A N ) A=(A_1,A_2,\dots,A_N) A=(A1,A2,,AN)
请按给出的顺序回答下列 Q Q Q 个问题。

k k k个查询按以下格式给出:

  • 首先,将 A i k A_{i_k} Aik改为 x k x_k xk。这一改动将带入后续查询。
  • 然后,打印 A A A m e x \rm{mex} mex
    • A A A m e x \rm{mex} mex是不包含在 A A A中的最小非负整数。

可以说一道典题,收获不小。有两种思路:第一种就是set,第二种就是树状数组,两种方法接下来都会介绍。

set做法
我们使用set去记录序列中没有出现过的数,那么这样对于每次查询而言,我们输出set的第一个元素即可。同时,我们使用 c n t cnt cnt数组去维护每个数在序列中的出现次数,然后每次修改时去 c n t cnt cnt数组中对应删除或是添加。
对于删除操作,若该元素删去一次后变为 0 ,即代表这个数不存在于序列中了,所以需要放入 set ;
对于添加操作,若该元素添加后次数变为1,即代表该数第一次出现在序列中(即之前不存在于序列中,存在于set中),所以此时需要把这个数从set中删除即可。
时间复杂度: q l o g n qlogn qlogn

#include <bits/stdc++.h>

using namespace std;
const int N = 2e5 + 5;
typedef long long ll;
typedef pair<ll, ll> pll;
typedef array<ll, 3> p3;
int mod = 1e9+7;
const int maxv = 4e6 + 5;
// #define endl "\n"



void solve()
{
	int n,q;
	cin>>n>>q;
	set<int> s;
	map<int,int >mp;//不使用map,使用正常数组即可。
	vector<int> a(n+5);
	for(int i=1;i<=n;i++){
		cin>>a[i];
		mp[a[i]]++;
	}
	for(int i=0;i<=n;i++){
		if(!mp.count(i)) s.insert(i);
	}
	while(q--){
		int id,x;
		cin>>id>>x;
		if(mp[a[id]]){
			mp[a[id]]--;
			mp[x]++;
			if(mp[x]==1) s.erase(x);
			if(mp[a[id]]==0){
				s.insert(a[id]);
			}
			a[id]=x;
		}
		cout<<*s.begin()<<endl;
	}
	

}

int main()
{
    ios::sync_with_stdio(0);
	cin.tie(0);
	cout.tie(0);
	int t;
	t=1;
	//cin>>t;
	while(t--){
		solve();
	}
   	system("pause");
    return 0;
}

树状数组
思路:我们考虑将每个元素放入树状数组中,即将值域映射到下标,通过0/1去判断这个数是否存在。经过上述操作后,我们对于mex的判断为:因为树状数组返回的是 1 − i 1-i 1i的前缀和,即对于第 i 个数,我们可以知道前面有多少个比 i 小的数,那么我们可以在树状数组上进行二分(前缀和保证了单调性),找到第一个下标大于其对应值的地方,即为mex。
时间复杂度: q × ( l o g n ) 2 q\times(logn)^2 q×(logn)2
细节有点多,具体看代码注释。

#include <bits/stdc++.h>

using namespace std;
const int N = 4e5 + 5;
typedef long long ll;
typedef pair<ll, ll> pll;
typedef array<ll, 3> p3;
int mod = 1e9+7;
const int maxv = 4e6 + 5;
// #define endl "\n"

int n,q;

vector<int> a(N),b(N);

struct MIT
{
ll tr[N];
int lowbit(int x) {
    return x & (-x);
}

void add(int u, int v) {
    for (int i = u; i < N; i += lowbit(i)) {
        tr[i] += v;
    }
}

ll query(int x) {
    ll res = 0;

    for (int i = x; i > 0; i -= lowbit(i)) {
        res += tr[i];
    }

    return res;
}
};
MIT c;

int cal()//树状数组二分过程
{
	int l=1,r=n+1,ans=0;
	while(l<=r){
		int mid=(l+r)/2;
		if(c.query(mid)==mid){
			ans=mid;
			l=mid+1;
		}
		else r=mid-1;
	}
	return ans;
}

void solve()
{
	cin>>n>>q;
	for(int i=1;i<=n;i++) cin>>a[i],a[i]++;//把每个元素加1,因为树状数组不能为0
	for(int i=1;i<=n;i++){
		if(a[i]<=n+1){//如果当前数大于n,那么放入树状数组就没有意义了,因为mex只能为0-n之间
			if(b[a[i]]==0){//如果这个数没有出现过,我们就进行添加
				c.add(a[i],1);
			}
			b[a[i]]++;//记录该数的出现次数
		}
	}
	while(q--){
		ll id,x;
		cin>>id>>x;
		x++;
		if(a[id]<=n+1){//同理
			if(b[a[id]]==1) c.add(a[id],-1);
			//如果当前数的次数为1,即删一次为0,即我们修改后这个数不存在了,所以就需要把这个数从树状数组中删除
			b[a[id]]--;
		}
		if(x<=n+1){//同理
			if(b[x]==0) c.add(x,1);
			b[x]++;
		}
		a[id]=x;//把当前数的值修改为x
		cout<<cal()<<endl;
	}
	

}

int main()
{
    ios::sync_with_stdio(0);
	cin.tie(0);
	cout.tie(0);
	int t;
	t=1;
	//cin>>t;
	while(t--){
		solve();
	}
   	system("pause");
    return 0;
}

由于set的做法写的不是很好,导致set的运行时间比树状数组还长…

大风起兮

在这里插入图片描述
简化一下题意,给定一个数组,有 q q q次操作,每次操作选择一个编号为 x x x的元素删除,要求输出每次操作的中位数。

思路
考虑树状数组在这题中如何进行应用,我们同样把值域映射到下标,去统计对于第 i 个数,其前面有多少个比他小的数。然后运用二分去找值为 n / 2 n/2 n/2的下标即为所求。

因为值域很大所以需要进行离散化。

#include <bits/stdc++.h>

using namespace std;
const int N = 2e5 + 5;
typedef long long ll;
typedef pair<ll, ll> pll;
typedef array<ll, 3> p3;
int mod = 1e9+7;
const int maxv = 4e6 + 5;
// #define endl "\n"

struct MIT
{
ll tr[N];
int lowbit(int x) {
    return x & (-x);
}

void add(int u, int v) {
    for (int i = u; i < N; i += lowbit(i)) {
        tr[i] += v;
    }
}

ll query(int x) {
    ll res = 0;

    for (int i = x; i > 0; i -= lowbit(i)) {
        res += tr[i];
    }

    return res;
}
};

int n,q;
int a[N];
MIT tr;

int cal(int x)
{
	int l=1,r=N,ans=0;
	while(l<=r){
		int mid=(l+r)/2;
		if(tr.query(mid)>=x){
			r=mid-1;
			ans=mid;
		}
		else l=mid+1;
	}
	return ans;
}

void solve()
{
	cin>>n;
	vector<int> b;
	for(int i=1;i<=n;i++){
		scanf("%d",&a[i]);
		b.push_back(a[i]);
	}
	b.push_back(0);
	sort(b.begin(),b.end());
	b.erase(unique(b.begin(),b.end()),b.end());//离散化不多说了
	for(int i=1;i<=n;i++){
		int t=lower_bound(b.begin(),b.end(),a[i])-b.begin();
		a[i]=t;
		tr.add(t,1);
	}
	int q;
	cin>>q;
	while(q--){
		int x;
		cin>>x;
		x=a[x];
		tr.add(x,-1);
		n--;
		if(n%2==0){
			double ans=(b[cal(n/2)]+b[cal(n/2+1)])*1.0/2;
			printf("%.1lf ",ans);
		}
		else{
			double ans=b[cal((n+1)/2)]*1.0;
			printf("%.1lf ",ans);
		}
	}
	//cout<<endl;
}

int main()
{
	int t;
	t=1;
	while(t--){
		solve();
	}
   	//system("pause");
    return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

找不到vcomp120.dll该如何修复?vcomp120.dll丢失的5个可行解决方法

本文将对vcomp120.dll文件的丢失原因进行详细分析&#xff0c;并提供五个有效的修复方法。同时&#xff0c;本文还将深入介绍vcomp120.dll文件的作用及其在程序运行中的重要性。 一、vcomp120.dll文件丢失原因 操作系统损坏&#xff1a;由于病毒感染、系统错误等原因&#xf…

linux复习笔记04(小滴课堂)

软件安装rpm方式介绍&#xff1a; 先去挂载光盘&#xff1a; 要确保这是已连接状态。 我们查看到已经挂载成功了。 进到这个目录下。 我们可以看到这有很多rpm软件包。 man rpm: 可以看到很多参数&#xff0c;但是我们不需要全部掌握。 举例&#xff1a; 这就是告诉我们需要安…

docker (简介、dcoker详细安装步骤)- day01

一、 为什么出现 Docker是基于Go语言实现的云开源项目。 Docker的主要目标是“Build&#xff0c;Ship and Run Any App,Anywhere”&#xff0c;也就是通过对应用组件的封装、分发、部署、运行等生命周期的管理&#xff0c;使用户的APP&#xff08;可以是一个WEB应用或数据库应…

WiFi的CSMA/CA竞争窗口流程简述

1、若站点最初有数据要发送&#xff08;不是发送不成功再进行重传的那种&#xff09;&#xff0c;且检测到信道空闲&#xff0c;在等待DIFS后&#xff0c;就发送整个数据帧。 2、否则&#xff0c;站点执行退避算法。一旦检测到信道忙&#xff0c;就冻结退避计时器。只要信道空…

深度学习之循环神经网络

视频链接&#xff1a;6 循环神经网络_哔哩哔哩_bilibili 给神经网络增加记忆能力 对全连接层而言&#xff0c;输入输出的维数固定&#xff0c;因此无法处理序列信息 对卷积层而言&#xff0c;因为卷积核的参数是共享的&#xff0c;所以卷积操作与序列的长度无关。但是因为卷积…

北塞浦路斯土耳其共和国关于成立欧洲数字股票交易所企业交流会

在地中海的温暖波涛中&#xff0c;北塞浦路斯土耳其共和国这个古老而充满活力的国家正成为全球关注的焦点。2023年11月22日至11月24日&#xff0c;为期三天的北塞浦路斯土耳其共和国关于成立欧洲数字股票交易所企业交流会隆重谢幕&#xff0c;北塞副总统&#xff0c;经济部长&a…

【学习记录】从0开始的Linux学习之旅——驱动模块编译与加载

一、概述 Linux操作系统通常是基于Linux内核&#xff0c;并结合GNU项目中的工具和应用程序而成。Linux操作系统支持多用户、多任务和多线程&#xff0c;具有强大的网络功能和良好的兼容性。本文主要讲述如何编译及加载linux驱动模块。 二、概念及原理 应用程序通过系统调用与内…

STK Components 二次开发-创建地面站

1.地面站只需要知道地面站的经纬高。 // Define the location of the facility using cartographic coordinates.var location new Cartographic(Trig.DegreesToRadians(-75.596766667), Trig.DegreesToRadians(40.0388333333), 0.0); 2.创建地面站 创建方式和卫星一样生成对…

【开源】基于Vue+SpringBoot的食品生产管理系统

项目编号&#xff1a; S 044 &#xff0c;文末获取源码。 \color{red}{项目编号&#xff1a;S044&#xff0c;文末获取源码。} 项目编号&#xff1a;S044&#xff0c;文末获取源码。 目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 加工厂管理模块2.2 客户管理模块2.3…

UE5人物残影学习(材质实现)

学习视频 UE4简单的材质球残影人教学&#xff0c;你学会了吗&#xff01;_哔哩哔哩_bilibilihttps://www.bilibili.com/video/BV1rY411q7Yb/?spm_id_from333.788.top_right_bar_window_history.content.click 结果预览 1.创建残值&#xff0c;混合模式勾选半透明 “混合模…

Qt4利用MVC开发曲线数据编辑器

目录 1 需求 2 开发流程 1 搭建框架 2 构造函数 3 打开工程 4 实现应用程序参数加载 5 QCustomPlot和TableView的联动 6 数据的可视化修改 7 列表点击事件事先键盘控制 8 表格实现复制&#xff0c;粘贴&#xff0c;删除等一系列功能 9 曲线实现自适应范围和统一范围…

MyBatis插入操作返回主键报错问题记录

一开始用直接传参数的方法写的插入操作 StudentMapper.java接口 Integer insertStudent(Param("sname") String name,Param("sage") int age); 然后在网上搜了返回主键的方法 StudentMapper.xml: <insert id"insertStudent" useGenerat…

CAN通信协议

CAN 文章目录 CAN前言一、什么是CAN二、CAN的用途三、CAN协议简解1.can的通信过程1.1空闲状态1.2.起始状态1.3 仲裁机制1.4 位时序 前言 前面学了232、485、IIC、SPI等通信协议&#xff0c;还有一个强大的协议CAN&#xff0c;值得记录一下 一、什么是CAN CAN是Controller Ar…

爬取极简壁纸

js反编译的代码需要解密之类的&#xff0c;直接给我干蒙圈了&#xff0c;借助selenium可以直接获取到调式工具中的源码&#xff0c;可以获取渲染后的链接&#xff0c;然后将链接交给下载函数&#xff08;使用异步提高效率&#xff09;即可。 后续学习完js反编译的话&#xff0…

Unity-链接MySql8.0

链接MySql8.0 1.准备dll 一、找到l18N相关的dll 这里给出一个参考地址 D:\Unity\2020.3.48f1c1\Editor\Data\MonoBleedingEdge\lib\mono\unityjit在里面找到如下图的四个dll 二、下载数据库链接dll https://downloads.mysql.com/archives/c-net/在这里搜索历史版本(Archiv…

AIGC ChatGPT4总结Linux Shell命令集合

在Linux中,Shell命令的数量非常庞大,因为Linux提供了各种各样的命令来处理系统任务。这些命令包括GNU核心工具集、系统命令、shell内置命令以及通过安装获得的第三方应用程序命令。以下是一些常见的Linux命令分类及其示例,但请注意,这不是一个全面的列表,因为列出所有命令…

linux复习笔记05(小滴课堂)

hell脚本与crontab定时器的运用 查看状态&#xff1a; 关闭服务&#xff1a; 开启服务&#xff1a; 重启服务&#xff1a; crontab定时器的使用&#xff1a; 我们可以看到没有任何任务。 编辑&#xff1a; 我们可以看到这个任务了。 删除所有任务&#xff1a; 这代表着每分钟…

【NeRF】3、MobileR2L | 移动端实时的神经光场(CVPR2023)

论文&#xff1a;Real-Time Neural Light Field on Mobile Devices 代码&#xff1a;https://github.com/snap-research/MobileR2L 出处&#xff1a;CVPR2023 贡献&#xff1a; 设计了一套移动端实时的 R2L 网络结构 MobileR2L&#xff0c;在 iphone13 上渲染一张 1008x756…

PgSQL技术内幕-Analyze做的那些事-pg_stat_all_tables

PgSQL技术内幕-Analyze做的那些事-pg_stat_all_tables pg_stat_all_tables视图中记录有analyze信息&#xff0c;比如何时做的analyze、表元组个数&#xff08;活元组、死元组&#xff09;等。重启后发现该视图中表的统计信息重置不见了&#xff0c;发生了什么&#xff1f; 1、p…

Centos 7.9 Install Docker Insecure Registry

文章目录 1. 镜像存储规划2. 安装定制 docker3. 部署 registry4. 验证镜像仓库 1. 镜像存储规划 linux LVM /dev/sdb mount dir /data【linux LVM 磁盘挂载目录】 创建两个目录 一个 docker 数据存储目录 &#xff1a;/data/docker&#xff0c;默认一般为linux为 /var/lib/d…