1058: [ZJOI2007]报表统计

Time Limit: 15 Sec  Memory Limit: 162 MB Submit: 4099  Solved: 1390 [Submit][Status][Discuss]

Description

  小Q的妈妈是一个出纳,经常需要做一些统计报表的工作。今天是妈妈的生日,小Q希望可以帮妈妈分担一些工
作,作为她的生日礼物之一。经过仔细观察,小Q发现统计一张报表实际上是维护一个可能为负数的整数数列,并
且进行一些查询操作。在最开始的时候,有一个长度为N的整数序列,并且有以下三种操作: INSERT i k 在原数
列的第i个元素后面添加一个新元素k; 如果原数列的第i个元素已经添加了若干元素,则添加在这些元素的最后(
见下面的例子) MIN_GAP 查询相邻两个元素的之间差值(绝对值)的最小值 MIN_SORT_GAP 查询所有元素中最接
近的两个元素的差值(绝对值) 例如一开始的序列为 5 3 1 执行操作INSERT 2 9将得到: 5 3 9 1 此时MIN_GAP
为2,MIN_SORT_GAP为2。 再执行操作INSERT 2 6将得到: 5 3 9 6 1 注意这个时候原序列的第2个元素后面已经
添加了一个9,此时添加的6应加在9的后面。这个时候MIN_GAP为2,MIN_SORT_GAP为1。于是小Q写了一个程序,使
得程序可以自动完成这些操作,但是他发现对于一些大的报表他的程序运行得很慢,你能帮助他改进程序么?

Input

  第一行包含两个整数N,M,分别表示原数列的长度以及操作的次数。第二行为N个整数,为初始序列。接下来
的M行每行一个操作,即“INSERT i k”,“MIN_GAP”,“MIN_SORT_GAP”中的一种(无多余空格或者空行)。

Output

  对于每一个“MIN_GAP”和“MIN_SORT_GAP”命令,输出一行答案即可。

Sample Input

3 5 5 3 1 INSERT 2 9 MIN_SORT_GAP INSERT 2 6 MIN_GAP MIN_SORT_GAP

Sample Output

2 2 1

HINT

N , M ≤500000 对于所有的数据,序列内的整数不超过5*10^8。

http://www.lydsy.com/JudgeOnline/problem.php?id=1058

                          - by bzoj



首先明确每次插入的位置可以通过简单的记录来快速得知,

然后分开考虑两种询问

1询问排序后的相邻元素间的最小差值,

如果维护排序后相邻元素间差值的话,元素的加入影响差值,应该是一个有加入有删除的操作,

而事实上由于每一次删除差值的诱因一定是新加入的元素夹在原来的两个元素之间,破坏了这两个元素间的差值,

这意味着删除的差值一定不如新产生的差值小,

于是可以视作没有删除;

那就是一个单纯比较取最小的过程了;

为了找到新产生了哪些差值,笔者这里选用了set对报表中的元素进行动态排序加二分查找;

这个询问的答案只会越来越小,最后可能到零,记得到零后记录一下,省一点多余的操作

2询问当前没有排序的序列的相邻元素的最小值,

如果维护未排序时相邻元素间差值的话,元素的加入影响差值,这里真的是一个有加入有删除的操作了;

每次在i与i+1间插入一个元素x,在维护差值的数据结构中找到abs(a[i+1]-a[i])删掉,然后加入abs(x-a[i])和abs(x-a[i+1]);

这里可以用平衡树维护

笔者这里偷懒直接用动态加点线段树做了,本来内存巨大应该过不了,遂偷偷改小数组,没想到竟然bzoj和luogu的数据都水过来了,暗爽

——现在写题解时想想,好像只要写个set维护就完事了

——结果就是道STL题啊!!!!!!!!!!!!

代码:

#include<map>
#include<cmath>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
map <int ,int >MAP;
int id_now[500010];
struct ss{
int num,las,nex;
}ctn[1000010];
struct DT{
int size,ch[2];
}data[10000000];
int top,tot,n,m;
bool flag;
char s[20];
inline void in(int &ans)
{
ans=0;bool p=false;char ch=getchar();
while((ch>'9' || ch<'0')&&ch!='-') ch=getchar();
if(ch=='-') p=true,ch=getchar();
while(ch<='9'&&ch>='0') ans=ans*10+ch-'0',ch=getchar();
if(p) ans=-ans;
}
void insert(int ,int ,int ,int ,int );
int get_MIN(int ,int ,int );
int main()
{
int i,j,k,ans2=1000000000;
map <int ,int > :: iterator iter;
in(n),in(m);
tot=n,top=1,flag=false;
for(i=1;i<=n;i++){
ctn[i].las=i-1;
ctn[i].nex=(i+1)%(n+1);
in(ctn[i].num);
if(ctn[i].las)
insert(0,1000000000,1,abs(ctn[i].num-ctn[ctn[i].las].num),1);
if(MAP.count(ctn[i].num)||flag)
flag=true;
else{
iter=MAP.upper_bound(ctn[i].num);
if(iter!=MAP.end())
ans2=min(ans2,iter->first-ctn[i].num);
if(iter!=MAP.begin())
iter--,ans2=min(ans2,ctn[i].num-iter->first);
MAP[ctn[i].num]=i;
}
id_now[i]=i;
}
for(i=1;i<=m;i++){
scanf("%s",s);
if(s[0]=='I'){
in(j),in(k);
ctn[++tot].num=k;
ctn[tot].nex=ctn[id_now[j]].nex;
ctn[ctn[id_now[j]].nex].las=tot;
ctn[tot].las=id_now[j];
ctn[id_now[j]].nex=tot;
ctn[0].nex=ctn[0].las=ctn[0].num=0;
if(ctn[tot].las)insert(0,1000000000,1,abs(k-ctn[ctn[tot].las].num),1);
if(ctn[tot].nex)insert(0,1000000000,1,abs(k-ctn[ctn[tot].nex].num),1);
if(ctn[tot].las&&ctn[tot].nex)insert(0,1000000000,1,abs(ctn[ctn[tot].las].num-ctn[ctn[tot].nex].num),-1);
if(!flag){
if(MAP.count(ctn[tot].num)){
flag=true,id_now[j]=tot;continue;
}
iter=MAP.upper_bound(k);
if(iter!=MAP.end())ans2=min(ans2,iter->first-k);
if(iter!=MAP.begin())iter--,ans2=min(ans2,k-iter->first);
MAP[k]=tot;
}
id_now[j]=tot;
}
if(s[0]=='M'&&s[4]=='G')
printf("%d\n",get_MIN(0,1000000000,1));
if(s[0]=='M'&&s[4]=='S')
printf("%d\n",(flag^1)*ans2);
}
return 0;
}
void insert(int l,int r,int now,int lim,int x){
int mid=(l+r)>>1;
if(l==r){
data[now].size+=x;
return ;
}
if(lim<=mid){
if(!data[now].ch[0])data[now].ch[0]=++top;
insert(l,mid,data[now].ch[0],lim,x);
}
else{
if(!data[now].ch[1])data[now].ch[1]=++top;
insert(mid+1,r,data[now].ch[1],lim,x);
}
data[now].size=data[data[now].ch[0]].size+data[data[now].ch[1]].size;
}
int get_MIN(int l,int r,int now){
if(l==r)return l;
int mid=(l+r)>>1;
if(data[data[now].ch[0]].size)
return get_MIN(l,mid,data[now].ch[0]);
else
return get_MIN(mid+1,r,data[now].ch[1]);
}

bzoj P1058 [ZJOI2007]报表统计——solution的更多相关文章

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

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

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

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

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

    Description 小Q的妈妈是一个出纳,经常需要做一些统计报表的工作.今天是妈妈的生日,小Q希望可以帮妈妈分担一些工 作,作为她的生日礼物之一.经过仔细观察,小Q发现统计一张报表实际上是维护一个 ...

  4. BZOJ 1058: [ZJOI2007]报表统计 multiset + 卡常

    Code: #include<bits/stdc++.h> #define maxn 600000 #define inf 1000000000 using namespace std; ...

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

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

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

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=1058 [题意] 一个序列,提供插入,查询相邻最小差值,查询任意最小差值的操作. [思路 ...

  7. bzoj 1058: [ZJOI2007]报表统计【set】

    我想写FHQtreap的!是set自己跑进代码的!因为太好写了 是有点慢--洛谷上不吸氧会T一个点 就是,用一个set p维护所有点值,ans维护MIN_SORT_GAP的答案,每次insert一个点 ...

  8. [ZJOI2007]报表统计(splay,堆)

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

  9. bzoj1058: [ZJOI2007]报表统计

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

随机推荐

  1. [Swift实际操作]七、常见概念-(11)路径URL的使用详解

    本文将为你演示网址对象(URL)的使用 首先导入需要使用的界面工具框架 import UIKit 接着初始化一个指定网址的网址对象 let url = URL(string: "https: ...

  2. jmeter-linux下运行

    1.2 在命令行下运行脚本 将1.1中的脚本保存,在编辑是随时可以保存,保存后是一个jmx格式的文件(如图),这个就是要在命令行下运行的脚本(作为参数运行).这个脚本文件可以不包含1.1中第四和第五步 ...

  3. 用postman做接口测试实例

    使用postman做接口测试,可以选择请求方式,可以直接输入参数和header,可以编写测试结果的代码,判断是否通过测试 下图为填写接口测试地址.填写接口的参数,点击send发送请求 其中,Param ...

  4. Supporting Right-to-Left Languages

    For the most part iOS supports Right-to-Left (RTL) languages such as Arabic with minimal developer e ...

  5. POJ 1006

    #include<iostream>#include<stdio.h>using namespace std;int main(){    //freopen("ac ...

  6. 大数据技术之_19_Spark学习_04_Spark Streaming 应用解析小结

    ========== Spark Streaming 是什么 ==========1.SPark Streaming 是 Spark 中一个组件,基于 Spark Core 进行构建,用于对流式进行处 ...

  7. java基本语法一

    1 关键字和保留字 1.1 关键字 关键字的定义:被java语言赋予了特殊含义,用做专门用途的字符串(单词). 关键字的特点:关键字中的所有字母都是小写. 1.2 保留字 java保留字:现有Java ...

  8. script全局变量

    javascript全局变量的生命周期是:直到页面关闭才失效.否则一直有用. 方式1 1 2 var test; var test = 5; 需注意的是该句不能包含在function内,否则是局部变量 ...

  9. Python 1行代码实现文本分类(实战笔记),含代码详细说明及运行结果

    Python 1行代码实现文本分类(实战笔记),含代码详细说明及运行结果 一.详细说明及代码 tc.py =============================================== ...

  10. Redis查询&JDBC查询&Hibernate查询方式的效率比较...

    比较三种查询方式查询效率对比...我是用的JavaWeb的方式通过通过JSP页面查询的填写查询的参数...给予反馈.... 整个demo的下载地址:http://files.cnblogs.com/f ...