Link:

BZOJ 1058 传送门

Solution:

为了这道题今天下午一直都在和常数大战……

1、对于询问1,我们记录每个数末位置的数$T[i]$和初始位置$S[i]$

用平衡树维护所有差值,对于操作$i,k$:删除$S[i+1]-T[i]$,增加$x-T[i]$和$x-S[i+1]$

2、对于询问2,用平衡树记录所有值,每次找到$k$的前驱和后继更新$mn$就好了

当然以上都可以不用手写平衡树,用$multiset$和$priority\_ queue$可水过

但我如此耿直的人还是用结构体写了两个$Splay$分别维护两个询问

下面总结下出的各种锅和常数优化细节吧:

1、补锅匠系列:

(1)范围要开到$2*n$,毕竟每次会新加进一个数

这时显示的居然是$TLE$不是$RE$?以后要注意……

(2)对于询问2在查找前驱后继时要包含与查找数相同的数

此时不是严格小于/大于啊……

2、卡常系列:

(1)基础的IO,$register$,$inline$

(2)进行任何操作后最好都$Splay$一遍来保证复杂度!

(3)如果$mn=0$时直接将后面的更新剪枝,好像对$Luogu$上的数据很有效……

(4)尽量少外界调用结构体内函数

卡常卡到最后还是只过了$Luogu$,$BZOJ$过不去啊……

但好像$BZOJ$数据加强了,黄学长的标程也T了……

最后发现别人用$Treap$写比我的快10倍?以后还是少写$Splay$吧

Code:

#include <bits/stdc++.h>

using namespace std;//注意MAXN要开到1e6
const int MAXN=1e6+,INF=0x3f3f3f3f;
char s[];int n,m,dat[MAXN],S[MAXN],T[MAXN],mn=INF; inline int read()
{
char ch;int num,f=;
while(!isdigit(ch=getchar())) f|=(ch=='-');
num=ch-'';
while(isdigit(ch=getchar())) num=num*+ch-'';
return f?-num:num;
}
inline void write(int x)
{
if(x<) putchar('-'),x=-x;
if(x>) write(x/);
putchar(x%+'');
} struct splay
{
int rt,sz[MAXN],cnt[MAXN],val[MAXN],f[MAXN],ch[MAXN][],tot;
inline void pushup(int x)
{sz[x]=sz[ch[x][]]+sz[ch[x][]]+cnt[x];}
inline void Rotate(int x)
{
int y=f[x],z=f[y],k=(ch[y][]==x);
ch[z][ch[z][]==y]=x;f[x]=z;
ch[y][k]=ch[x][k^];f[ch[x][k^]]=y;
ch[x][k^]=y;f[y]=x;
pushup(x);pushup(y);
}
inline void Splay(int x,int up)
{
while(f[x]!=up)
{
int y=f[x],z=f[y];
if(z!=up) (ch[y][]==x)^(ch[z][]==y)?Rotate(x):Rotate(y);
Rotate(x);
}
if(!up) rt=x;
}
inline void Insert(int x)
{
int k=rt,anc=;
while(k&&x!=val[k])
anc=k,k=ch[k][x>val[k]]; if(k) {cnt[k]++;Splay(k,);return;}
k=++tot;
if(anc) ch[anc][x>val[anc]]=k;
ch[k][]=ch[k][]=;
sz[k]=cnt[k]=;f[k]=anc;val[k]=x;
Splay(k,);
}
inline void Find(int x)
{
int k=rt;
while(ch[k][x>val[k]]&&x!=val[k])
k=ch[k][x>val[k]];
Splay(k,);
}
inline int Kth(int x)
{
int k=rt;
while(true)
{
if(sz[ch[k][]]+cnt[k]<x)
x-=sz[ch[k][]]+cnt[k],k=ch[k][];
else if(sz[ch[k][]]>=x) k=ch[k][];
else return k;
}
}
inline int Next(int x,int flag)
{
Find(x);int k=rt;
if((val[k]<x&&!flag)||(val[k]>x&&flag)) return k;
k=ch[k][flag];
while(ch[k][flag^]) k=ch[k][flag^];
Splay(k,);return k;
}
inline int Next2(int x,int flag)
{
Find(x);int k=rt;
if((val[k]<=x&&!flag)||(val[k]>=x&&flag)) return k;
k=ch[k][flag];
while(ch[k][flag^]) k=ch[k][flag^];
Splay(k,);return k;
}
inline void Delete(int x)
{
int nxt=Next(x,),lst=Next(x,);
Splay(lst,);Splay(nxt,lst);
if(cnt[ch[nxt][]]>)
cnt[ch[nxt][]]--,Splay(ch[nxt][],);
else ch[nxt][]=;
}
inline void push(int x)
{//调用结构体函数的次数越少越好,因此push放里面快
if(!mn) return;
int nxt=Next2(x,),lst=Next2(x,);
mn=min(mn,min(abs(val[nxt]-x),abs(val[lst]-x)));
Insert(x);
}
}all,adj; int main()
{
n=read();m=read();
all.Insert(INF);all.Insert(-INF);
adj.Insert(INF);adj.Insert(-INF);
for(register int i=;i<=n;i++)
dat[i]=read(),all.push(dat[i]),S[i]=T[i]=dat[i];
for(register int i=;i<=n;i++) adj.Insert(abs(dat[i]-dat[i-])); while(m--)
{
int x,y;scanf("%s",s);
if(s[]=='I')
{
x=read();y=read();
if(x!=n) adj.Delete(abs(S[x+]-T[x]));
adj.Insert(abs(y-T[x]));
adj.Insert(abs(y-S[x+]));
all.push(y);T[x]=y;
}
else if(s[]=='G') write(adj.val[adj.Kth()]),putchar('\n');
else write(mn),putchar('\n');
}
return ;
}

[BZOJ 1058] 报表统计的更多相关文章

  1. BZOJ 1058 报表统计 (STL)

    题解:数据结构的基本操作,用STL可以完美实现,就是比较慢…… #include <cstdio> #include <map> #include <set> #i ...

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

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

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

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

  4. [BZOJ 1058] [ZJOI2007] 报表统计 【平衡树】

    题目链接:BZOJ - 1058 题目分析 这道题看似是需要在序列中插入一些数字,但其实询问的内容只与相邻的元素有关. 那么我们只要对每个位置维护两个数 Ai, Bi, Ai 就是初始序列中 i 这个 ...

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

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

  6. bzoj1058: [ZJOI2007]报表统计

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

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

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

  8. 【BZOJ1058】【ZJOI2007】报表统计(链表,堆,Splay)

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

  9. BZOJ_1058_[ZJOI2007]报表统计_STL

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

随机推荐

  1. Spring整合Quartz分布式调度(山东数漫江湖)

    前言 为了保证应用的高可用和高并发性,一般都会部署多个节点:对于定时任务,如果每个节点都执行自己的定时任务,一方面耗费了系统资源,另一方面有些任务多次执行,可能引发应用逻辑问题,所以需要一个分布式的调 ...

  2. hadoop+spark 集群的安装

    1.安装连接 https://www.cnblogs.com/zengxiaoliang/p/6478859.html

  3. js常用模板引擎

    baiduTemplate(百度).artTemplate(腾讯).juicer(淘宝).xtemplate.doT.Jade 1.Handlebars 是 JavaScript 一个语义模板库,通过 ...

  4. [Leetcode Week16]Insertion Sort List

    Insertion Sort List 题解 原创文章,拒绝转载 题目来源:https://leetcode.com/problems/insertion-sort-list/description/ ...

  5. FAN54015 充電電流 軟硬體設定

    Ex1: Vrsense 選 37.4 mV --- 在第二張圖 Rsense 選 50 mΩ --- 在第三張圖 37.4 / 50 = 748 mA Ex2: Vrsense 選 44.2 mV ...

  6. 自动化测试===Macaca环境搭建,自我总结

    安装jdk 安装安卓sdk(打开sdk的时候出现问题linux===启动sdk manager下载配置sdk的时候报错的解决办法) 安装gradle,配置环境变量(MACACA===gradle下载和 ...

  7. (二十)ubuntu的recovery mode解决用户一些实际问题

    遇到的问题如下: 1.在当前用户下使用sudo来直接修改password等几个文件,一旦修改了passwd,用户名发生了变化,其他的用户组.密码等却没有对应的配置,就再进不了该用户了. 2.忘记用户密 ...

  8. 算法题之找出数组里第K大的数

    问题:找出一个数组里面前K个最大数. 解法一(直接解法): 对数组用快速排序,然后直接挑出第k大的数.这种方法的时间复杂度是O(Nlog(N)).N为原数组长度. 这个解法含有很多冗余,因为把整个数组 ...

  9. VPS性能测试(2):内存大小、交换空间、高速缓存、实际使用内存

    1.要想查看购买的VPS主机的内存信息,执行:cat /proc/meminfo,主要是看内存大小.交换空间.高速缓存 2.VPS主机实际使用内存大小.Linux管理内存的机制是这样的:无论物理内存有 ...

  10. Mac iphone 使用 如何修改apple 用户名 XXX的mac Mac 与iphone如何连接 传递文件 为iphone增加铃声 iphone铃声的制作---城

    1.更改mac apple id Apple ID 即用户名称,您可以将其用于与 Apple 有关的所有操作.为某个 Apple 服务(如 iCloud 或 App Store)创建帐户时即创建了 A ...