容易想到树套树,但数据范围太大,会超时
考虑平衡树的作用,就是将这个区间内的所有数排序,所以可以离线+归并来处理,预处理复杂度$o(n\log n)$,然后考虑维护:1.删除;2.询问
删除操作维护可以使用并查集,可以通过$\alpha(n)$的时间里快速找到每一个点的上和下元素
询问操作可以二分查找,这样的复杂度为$o(q\log^{2} n+q\log n\cdot \alpha(n))$,考虑优化掉二分的复杂度:对于每一个数,合并时记录左右区间中比他小/大且最接近的数,这样只需要在最开始二分一次就可以传递下来,复杂度降为$o(q\log n\cdot \alpha(n))$
实现中可能有一些细节问题:1.$nex[k].fi.fi$表示了左子树中小于等于它的最大数,若不存在则为最小的数,若没有数就随便(在query中要判断这个区间是否存在数,而且有可能因为删除而没有数),其余同理;2.每一个数要存储个数,否则会导致存在但无法找到;3.如果最小或最大的数删掉,并查集就指向不存在的位置来表示

  1 #include<bits/stdc++.h>
2 using namespace std;
3 #define N 1000005
4 #define mid (l+r>>1)
5 #define L (k<<1)
6 #define R (L+1)
7 #define mp make_pair
8 #define fi first
9 #define se second
10 #define oo 0x3f3f3f3f
11 struct ji{
12 int p,l,r,k;
13 }q[N];
14 vector<int>f[N<<1],tot[N<<1],fa[2][N<<1];
15 vector<pair<pair<int,int>,pair<int,int> > >nex[N<<1];
16 int n,m,x,y,ans[N],sz[N<<1],now[N<<1];
17 int find(int p,int x,int k){
18 if ((k==sz[x])||(k==fa[p][x][k]))return k;
19 return fa[p][x][k]=find(p,x,fa[p][x][k]);
20 }
21 void add(int k,int l,int r,int x,int y){
22 now[k]++;
23 if (l==r){
24 sz[k]=1;
25 f[k].push_back(y);
26 tot[k].push_back(1);
27 return;
28 }
29 if (x<=mid)add(L,l,mid,x,y);
30 else add(R,mid+1,r,x,y);
31 }
32 void merge(int k,int l,int r){
33 if (l==r){
34 if (!sz[k])return;
35 fa[0][k].push_back(0);
36 fa[1][k].push_back(0);
37 return;
38 }
39 merge(L,l,mid);
40 merge(R,mid+1,r);
41 for(int i=0,j=0;(i<sz[L])||(j<sz[R]);){
42 fa[0][k].push_back(sz[k]);
43 fa[1][k].push_back(sz[k]++);
44 if ((i<sz[L])&&(j<sz[R])&&(f[L][i]==f[R][j])){
45 nex[k].push_back(mp(mp(i,i),mp(j,j)));
46 f[k].push_back(f[L][i]);
47 tot[k].push_back(tot[L][i++]+tot[R][j++]);
48 continue;
49 }
50 if ((i<sz[L])&&((j==sz[R])||(f[L][i]<f[R][j]))){
51 nex[k].push_back(mp(mp(i,i),mp(j-(j>0),j-((j>0)&&(j==sz[R])))));
52 f[k].push_back(f[L][i]);
53 tot[k].push_back(tot[L][i++]);
54 }
55 else{
56 nex[k].push_back(mp(mp(i-(i>0),i-((i>0)&&(i==sz[L]))),mp(j,j)));
57 f[k].push_back(f[R][j]);
58 tot[k].push_back(tot[R][j++]);
59 }
60 }
61 }
62 void del(int k,int l,int r,int x,int y){
63 now[k]--;
64 if (--tot[k][y]==0){
65 if (!y)fa[0][k][y]=sz[k];
66 else fa[0][k][y]=fa[0][k][y-1];
67 if (y==sz[k]-1)fa[1][k][y]=sz[k];
68 else fa[1][k][y]=fa[1][k][y+1];
69 }
70 if (l==r)return;
71 if (x<=mid)del(L,l,mid,x,nex[k][y].fi.fi);
72 else del(R,mid+1,r,x,nex[k][y].se.fi);
73 }
74 int query(int k,int l,int r,int x,int y,int z,int a,int b){
75 if ((l>y)||(x>r)||(!now[k]))return oo;
76 if ((x<=l)&&(r<=y)){
77 int ans=oo;
78 if (find(0,k,a)<sz[k])ans=min(ans,abs(z-f[k][find(0,k,a)]));
79 if (find(1,k,b)<sz[k])ans=min(ans,abs(z-f[k][find(1,k,b)]));
80 return ans;
81 }
82 return min(query(L,l,mid,x,y,z,nex[k][a].fi.fi,nex[k][b].fi.se),query(R,mid+1,r,x,y,z,nex[k][a].se.fi,nex[k][b].se.se));
83 }
84 int main(){
85 scanf("%d%d",&n,&m);
86 for(int i=1;i<=m;i++){
87 scanf("%d%d",&q[i].p,&q[i].l);
88 if (q[i].p)scanf("%d%d",&q[i].r,&q[i].k);
89 else{
90 scanf("%d",&q[i].k);
91 add(1,1,n,q[i].l,q[i].k);
92 }
93 }
94 merge(1,1,n);
95 for(int i=m;i;i--)
96 if (!q[i].p)del(1,1,n,q[i].l,lower_bound(f[1].begin(),f[1].end(),q[i].k)-f[1].begin());
97 else{
98 x=upper_bound(f[1].begin(),f[1].end(),q[i].k)-f[1].begin()-1;
99 y=lower_bound(f[1].begin(),f[1].end(),q[i].k)-f[1].begin();
100 ans[++ans[0]]=query(1,1,n,q[i].l,q[i].r,q[i].k,max(x,0),y);
101 }
102 for(int i=ans[0];i;i--)
103 if (ans[i]==oo)printf("-1\n");
104 else printf("%d\n",ans[i]);
105 }

[noi1774]array的更多相关文章

  1. javascript中的Array对象 —— 数组的合并、转换、迭代、排序、堆栈

    Array 是javascript中经常用到的数据类型.javascript 的数组其他语言中数组的最大的区别是其每个数组项都可以保存任何类型的数据.本文主要讨论javascript中数组的声明.转换 ...

  2. ES5对Array增强的9个API

    为了更方便的对Array进行操作,ES5规范在Array的原型上新增了9个方法,分别是forEach.filter.map.reduce.reduceRight.some.every.indexOf ...

  3. JavaScript Array对象

    介绍Js的Array 数组对象. 目录 1. 介绍:介绍 Array 数组对象的说明.定义方式以及属性. 2. 实例方法:介绍 Array 对象的实例方法:concat.every.filter.fo ...

  4. 了解PHP中的Array数组和foreach

    1. 了解数组 PHP 中的数组实际上是一个有序映射.映射是一种把 values 关联到 keys 的类型.详细的解释可参见:PHP.net中的Array数组    . 2.例子:一般的数组 这里,我 ...

  5. 关于面试题 Array.indexof() 方法的实现及思考

    这是我在面试大公司时碰到的一个笔试题,当时自己云里雾里的胡写了一番,回头也曾思考过,最终没实现也就不了了之了. 昨天看到有网友说面试中也碰到过这个问题,我就重新思考了这个问题的实现方法. 对于想进大公 ...

  6. javascript之活灵活现的Array

    前言 就如同标题一样,这篇文章将会灵活的运行Array对象的一些方法来实现看上去较复杂的应用. 大家都知道Array实例有这四个方法:push.pop.shift.unshift.大家也都知道 pus ...

  7. 5.2 Array类型的方法汇总

    所有对象都具有toString(),toLocaleString(),valueOf()方法. 1.数组转化为字符串 toString(),toLocaleString() ,数组调用这些方法,则返回 ...

  8. OpenGL ES: Array Texture初体验

    [TOC] Array Texture这个东西的意思是,一个纹理对象,可以存储不止一张图片信息,就是说是是一个数组,每个元素都是一张图片.这样免了频繁地去切换当前需要bind的纹理,而且可以节省系统资 ...

  9. Merge Sorted Array

    Given two sorted integer arrays nums1 and nums2, merge nums2 into nums1 as one sorted array. Note:Yo ...

随机推荐

  1. 使用python -m pip install 和 pip install 安装包有什么区别?

    如果去看源码的话,你会发现 pip 作为执行文件的入口点是 pip._internal.main. 另一方面,pip 作为模块运行时入口是 _main.py,而该模块也只是调用 pip.interna ...

  2. JS 实现计算器功能

    括号功能未实现,后续更 <!DOCTYPE html> <html lang="en"> <head> <meta charset=&qu ...

  3. 给力!斩获 GitHub 14000 Star,两周创办开源公司获数百万美元融资

    文章来源|AI科技大本营 作者|伍杏玲 上世纪 90 年代初,21 岁大学生 Linus Torvalds 开源 Linux 操作系统,自此掀起全球开源浪潮.随后"中国 Linux 第一人& ...

  4. 前端必会的Javascript经典面试题

    这是前端最基础的问题,也是在面试中难倒无数同学的经典问题 01. Javascript 数据类型 Javascript 数据类型 = 基本类型 + 引用类型 ES6 之前 5 + 1 = 6 种 ES ...

  5. Johnson 全源最短路径算法学习笔记

    Johnson 全源最短路径算法学习笔记 如果你希望得到带互动的极简文字体验,请点这里 我们来学习johnson Johnson 算法是一种在边加权有向图中找到所有顶点对之间最短路径的方法.它允许一些 ...

  6. HCIP-RSTP

    端口角色 根端口(RP): 非根桥收到最优的BPDU配置信息的端口为根端口,(到根桥开销最小的端口),根桥没有根端口. 指定端口(DP): 两非根交换机之间连接的两个端口必有一个为指定端口,比较两个非 ...

  7. SpringCloud 2020.0.4 系列之 Feign

    1. 概述 老话说的好:任何问题都有不止一种的解决方法,当前的问题没有解决,只是还没有发现解决方法,而并不是无解. 言归正传,之前我们聊了 SpringCloud 的服务治理组件 Eureka,今天我 ...

  8. 面试题系列:new String("abc")创建了几个对象

    new String("abc")创建了几个对象 面试官考察点猜想 这种问题,考察你对JVM的理解程度.涉及到常量池.对象内存分配等问题. 涉及背景知识详解 在分析这个问题之前,我 ...

  9. Spring Authorization Server的使用

    Spring Authorization Server的使用 一.背景 二.前置知识 三.需求 四.核心代码编写 1.引入授权服务器依赖 2.创建授权服务器用户 3.创建授权服务器和客户端 五.测试 ...

  10. Go 里的超时控制

    前言 日常开发中我们大概率会遇到超时控制的场景,比如一个批量耗时任务.网络请求等:一个良好的超时控制可以有效的避免一些问题(比如 goroutine 泄露.资源不释放等). Timer 在 go 中实 ...