思路

这里的初始化就不讲了,看完操作讲解就应该明白了,再不行就去看代码

对于操作1

由于操作2的需要,vector[n]存下数

对于操作2的维护

查询相邻两个元素的之间差值(绝对值)的最小值

先把所有答案存入一个小头堆里

比如 a,c之间你要插入b

那么,你就要删除|c-a|,然后加入|a-b|,|c-b|

之后的堆顶就是ans啦

对于支持删除的小头堆,我只会fhqtreap代替

但我也不想写, 因为太麻烦了

那么,我们能不能用STL自带的priority_queue

当然是可行的

维护两个堆,x,y

ans存入x中,需要删除的存入y中

每次更改的时候,改存的放入x,改删的放入y

每当x.top()==y.top(),就一起删掉,(注意!y.empty())

对于操作3的维护

查询所有元素中最接近的两个元素的差值(绝对值)

每次修改一个数,对最优答案都不会有坏的影响(显然)

那好的影响是啥

就是他和她的前驱,后继的差(显然)

注意前驱,后继的存在性

错误及其优化

最近老是不动脑子了(或者根本没脑子?)

insert操作查找前驱后继的时候可以省去find操作

代码

#include <iostream>
#include <algorithm>
#include <cstdio>
#include <cstring>
#include <vector>
#include <queue>
#define FOR(i,a,b) for(int i=a;i<=b;++i)
using namespace std;
const int maxn=5e5+7;
const int inf=0x3f3f3f3f;
inline int read() {
int x=0,f=1;char s=getchar();
for(; s>'9'||s<'0'; s=getchar()) if(s=='-') f=-1;
for(; s>='0'&&s<='9'; s=getchar()) x=x*10+s-'0';
return x*f;
}
int n,m,rt,ans2=inf,a[maxn],cnt;
struct node {
int ch[2],val,fa,siz;
} e[maxn*2];
vector<int> x[maxn];
priority_queue<int, vector<int>, greater<int> > ans1,delet;
inline int abs(int a) {return a>0?a:-a;}
inline void pushup(int x) {e[x].siz=e[e[x].ch[0]].siz+e[e[x].ch[1]].siz+1;}
inline void rotate(int x) {
int y=e[x].fa,z=e[y].fa,k=e[y].ch[1]==x;
e[x].fa=z;
e[z].ch[e[z].ch[1]==y]=x;
e[y].ch[k]=e[x].ch[k^1];
e[e[x].ch[k^1]].fa=y;
e[y].fa=x;
e[x].ch[k^1]=y;
pushup(x);pushup(y);
}
void splay(int x,int goal) {
while(e[x].fa!=goal) {
int y=e[x].fa,z=e[y].fa;
if(z!=goal)(e[y].ch[1]==x)^(e[z].ch[1]==y) ? rotate(y):rotate(x);
rotate(x);
}
if(!goal) rt=x;
}
void find(int a) {
int now=rt;
while(e[now].ch[e[now].val<a]&&e[now].val!=a)
now=e[now].ch[e[now].val<a];
splay(now,0);
}
void insert(int x) {
int now=rt,fa=0;
while(now&&e[now].val!=x)
fa=now,now=e[now].ch[e[now].val<x];
now=++cnt;
if(fa) e[fa].ch[e[fa].val<x]=now;
e[now].val=x;
e[now].fa=fa;
e[now].siz=1;
splay(now,0);
}
inline int qq(int x) {
if(e[rt].val<x) return rt;
int now=e[rt].ch[0];
while(e[now].ch[1]) now=e[now].ch[1];
return now;
}
inline int hj(int x) {
if(e[rt].val>x) return rt;
int now=e[rt].ch[1];
while(e[now].ch[0]) now=e[now].ch[0];
return now;
}
int build(int l,int r,int fa) {
if(l>r) return 0;
int mid=(l+r)>>1,p=++cnt;
e[p].val=a[mid];
e[p].fa=fa;
e[p].siz=1;
e[p].ch[0]=build(l,mid-1,p);
e[p].ch[1]=build(mid+1,r,p);
pushup(p);
return p;
}
int main() {
n=read(),m=read();
FOR(i,1,n) a[i]=read(),x[i].push_back(a[i]);
//ans1
FOR(i,2,n) ans1.push(abs(x[i][0]-x[i-1][0]));
//ans2
sort(a+1,a+1+n);
FOR(i,2,n) ans2=min(abs(a[i]-a[i-1]),ans2);
a[0]=-inf,a[n+1]=inf;
rt=build(0,n+1,0);
char s[10];
FOR(i,1,m) {
scanf("%s",s);
if(s[4]=='R') {
int a=read(),b=read();
//ans1
if(a!=n) delet.push(abs(x[a][x[a].size()-1]-x[a+1][0]));
x[a].push_back(b);
if(a!=n) ans1.push(abs(x[a][x[a].size()-1]-x[a+1][0]));
ans1.push(abs(x[a][x[a].size()-1]-x[a][x[a].size()-2]));
while(delet.size()&&ans1.top()==delet.top()) ans1.pop(),delet.pop();
//ans2
find(b);
if(e[rt].val==b) {ans2=0;continue;}
insert(b);
int x=e[qq(b)].val,y=e[hj(b)].val;
ans2=min(ans2,min(abs(x-b),abs(y-b)));
} else if(s[4]=='G') {
cout<<ans1.top()<<"\n";
} else if(s[4]=='S') {
cout<<ans2<<"\n";
}
}
return 0;
}

luogu1110[ZJOI2007]报表统计的更多相关文章

  1. [luogu1110][ZJOI2007]报表统计【平衡树】

    传送门 [洛谷传送门] [bzoj传送门] 前言 洛谷和网上的题解都好复杂哦,或者是stl水过. 窝的语文不怎么好,所以会有一些表达上的累赘或者是含糊不清,望各大佬海涵. 前置芝士 首先你一定要会平衡 ...

  2. bzoj1058: [ZJOI2007]报表统计

    set.操作:insert(u,v)在u后面插入v,若u后面已插入过,在插入过的后面插入.mingap求出序列两两之间差值的最小值.minsortgap求出排序后的序列两两之间的最小值.用multis ...

  3. BZOJ 1058: [ZJOI2007]报表统计( 链表 + set )

    这种题用数据结构怎么写都能AC吧...按1~N弄个链表然后每次插入时就更新答案, 用set维护就可以了... --------------------------------------------- ...

  4. [补档][ZJOI2007] 报表统计

    [ZJOI2007] 报表统计 题目 传送门 小Q的妈妈是一个出纳,经常需要做一些统计报表的工作.今天是妈妈的生日,小Q希望可以帮妈妈分担一些工作,作为她的生日礼物之一. 经过仔细观察,小Q发现统计一 ...

  5. BZOJ_1058_[ZJOI2007]报表统计_STL

    BZOJ_1058_[ZJOI2007]报表统计_STL Description 小Q的妈妈是一个出纳,经常需要做一些统计报表的工作.今天是妈妈的生日,小Q希望可以帮妈妈分担一些工 作,作为她的生日礼 ...

  6. bzoj 1058: [ZJOI2007]报表统计 (Treap)

    链接:https://www.lydsy.com/JudgeOnline/problem.php?id=1058 题面; 1058: [ZJOI2007]报表统计 Time Limit: 15 Sec ...

  7. 洛谷 P1110 [ZJOI2007]报表统计 解题报告

    P1110 [ZJOI2007]报表统计 题目描述 \(Q\)的妈妈是一个出纳,经常需要做一些统计报表的工作.今天是妈妈的生日,小\(Q\)希望可以帮妈妈分担一些工作,作为她的生日礼物之一. 经过仔细 ...

  8. bzoj P1058 [ZJOI2007]报表统计——solution

    1058: [ZJOI2007]报表统计 Time Limit: 15 Sec  Memory Limit: 162 MB Submit: 4099  Solved: 1390 [Submit][St ...

  9. 【BZOJ1058】[ZJOI2007]报表统计 STL

    [BZOJ1058][ZJOI2007]报表统计 Description 小Q的妈妈是一个出纳,经常需要做一些统计报表的工作.今天是妈妈的生日,小Q希望可以帮妈妈分担一些工作,作为她的生日礼物之一.经 ...

随机推荐

  1. notepad使用列选

    列选有两种方法: 1.按住ALT + 鼠标从某点按住开始向下或向上拖动. 2.按住ALT+SHIFT+上下方向键. 列编辑: 1.ALT+C 2.插入相同文本还是自增数字

  2. 通过改hosts访问wikipedia

    能访问https://www.wikipedia.org 大部分时候我们是可以访问的wikipedia的主页或是英文首页的,但中文页面却经常被墙,这时候你可以通过在命令行下ping www.wikip ...

  3. 18. 4Sum(双指针)

    Given an array nums of n integers and an integer target, are there elements a, b, c, and d in nums s ...

  4. html5-fieldset和legend和keygen元素的用法

    <!DOCTYPE html><html lang="en"><head>    <meta charset="UTF-8&qu ...

  5. Block 循环引用(中)

    不会造成循环引用的block 大部分GCD方法 dispatch_async(dispatch_get_main_queue(), ^{ [self doSomething]; }); 因为self并 ...

  6. XML系列之--创建电文格式的XML(一)

    关于XML,学校那会,老师在口中仅仅提及,自己也未曾深入接触过,仅是些将最基本XML文件内容显示在web定义的表格中之类的简单操作,如今项目中的收发电文涉及到复杂XML的操作.趁此契机好好回顾下XML ...

  7. python SQLite说一点点, python使用数据库需要注意的几点

    SQLite是一种嵌入式数据库,它的数据库就是一个文件.由于SQLite本身是C写的,而且体积很小,所以,经常被集成到各种应用程序中,甚至在iOS和Android的App中都可以集成. Python就 ...

  8. python os.path.dirname()

    ----返回文件所在的路径 ----如果path变量直接是文件名则返回空

  9. linux常用命令:cp 命令

    cp命令用来复制文件或者目录,是Linux系统中最常用的命令之一.一般情况下,shell会设置一个别名,在命令行下复制文件时,如果目标文件已经存在,就会询问是否覆盖,不管你是否使用-i参数.但是如果是 ...

  10. mysql安装使用

    linux系统 mysql-5.7.14-linux.zip部署包支持在CentOS 6.x/7.x 服务器硬盘大小要求 a) /data/mysql_data  如果存在该独立分区,要求该分区 &g ...