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. Java并发工具类之CountDownLatch

    CountDownLatch允许一个或则多个线程等待其他线程完成操作. 假如我们有这样的需求:我们需要解析一个excel文件中的多个sheet,我们可以考虑使用多线程,每一个线程解析excel中的一个 ...

  2. [AIR] AIR程序调用本地默认应用程序打开本地文件

    摘要:      File类提供了一个方法openWithDefaultApplication可以用本地默认应用程序打开指定路径下的文件. 当我用下面语句的时候,可以成功打开桌面文件夹下面的文件: v ...

  3. 【STM32H7教程】第14章 STM32H7的电源,复位和时钟系统

    完整教程下载地址:http://forum.armfly.com/forum.php?mod=viewthread&tid=86980 第14章       STM32H7的电源,复位和时钟系 ...

  4. Ubuntu14.04安装之后的一些配置

    不多说,直接上干货! 主要分为 一.root用户的开启和vim编辑器的安装 二.ssh的安装 三.静态ip的设置 四.中英切换文环境切换 一.root用户的开启和vim编辑器的安装 Ubuntu在默认 ...

  5. XMind *思维导图的安装步骤(图文详解)

    不多说,直接上干货! XMind中文官网:  http://www.xmindchina.net/ 这一款软件,是非常实用和棒,也帮助我了很多地方.推荐给大家 需要正版和激活的,请见博文最下端的QQ技 ...

  6. ELK日志系统之通用应用程序日志接入方案

    前边有两篇ELK的文章分别介绍了MySQL慢日志收集和Nginx访问日志收集,那么各种不同类型应用程序的日志该如何方便的进行收集呢?且看本文我们是如何高效处理这个问题的 日志规范 规范的日志存放路径和 ...

  7. maven私服配置

    1.maven私服setting.xml的配置 <?xml version="1.0" encoding="UTF-8"?> <setting ...

  8. CSS控制字体在一行内显示不换行

    当一行文字超过DIV或者Table的宽度的时候,浏览器中默认是让它换行显示的,如果不想让他换行要怎么办呢?用CSS让文字在一行内显示不换行的方法: 一般的文字截断(适用于内联与块): .text-ov ...

  9. JavaScript引用类型和值类型

    thead>tr>th,.table>tbody>tr>th,.table>tfoot>tr>th,.table>thead>tr>t ...

  10. C#语法之扩展

    扩展方法使你能够向现有类型“添加”方法,而无需创建新的派生类型.重新编译或以其他方式修改原始类型. 扩展方法是一种特殊的静态方法,但可以像扩展类型上的实例方法一样进行调用.这是msdn的描述.上面几句 ...