bzoj P1058 [ZJOI2007]报表统计——solution
1058: [ZJOI2007]报表统计
Time Limit: 15 Sec Memory Limit: 162 MB Submit: 4099 Solved: 1390 [Submit][Status][Discuss]
Description
Input
Output
对于每一个“MIN_GAP”和“MIN_SORT_GAP”命令,输出一行答案即可。
Sample Input
Sample Output
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的更多相关文章
- BZOJ 1058: [ZJOI2007]报表统计( 链表 + set )
这种题用数据结构怎么写都能AC吧...按1~N弄个链表然后每次插入时就更新答案, 用set维护就可以了... --------------------------------------------- ...
- bzoj 1058: [ZJOI2007]报表统计 (Treap)
链接:https://www.lydsy.com/JudgeOnline/problem.php?id=1058 题面; 1058: [ZJOI2007]报表统计 Time Limit: 15 Sec ...
- bzoj 1058: [ZJOI2007]报表统计
Description 小Q的妈妈是一个出纳,经常需要做一些统计报表的工作.今天是妈妈的生日,小Q希望可以帮妈妈分担一些工 作,作为她的生日礼物之一.经过仔细观察,小Q发现统计一张报表实际上是维护一个 ...
- BZOJ 1058: [ZJOI2007]报表统计 multiset + 卡常
Code: #include<bits/stdc++.h> #define maxn 600000 #define inf 1000000000 using namespace std; ...
- [BZOJ 1058] [ZJOI2007] 报表统计 【平衡树】
题目链接:BZOJ - 1058 题目分析 这道题看似是需要在序列中插入一些数字,但其实询问的内容只与相邻的元素有关. 那么我们只要对每个位置维护两个数 Ai, Bi, Ai 就是初始序列中 i 这个 ...
- bzoj 1058 [ZJOI2007]报表统计(set)
[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=1058 [题意] 一个序列,提供插入,查询相邻最小差值,查询任意最小差值的操作. [思路 ...
- bzoj 1058: [ZJOI2007]报表统计【set】
我想写FHQtreap的!是set自己跑进代码的!因为太好写了 是有点慢--洛谷上不吸氧会T一个点 就是,用一个set p维护所有点值,ans维护MIN_SORT_GAP的答案,每次insert一个点 ...
- [ZJOI2007]报表统计(splay,堆)
[ZJOI2007]报表统计(luogu) Description 题目描述 Q的妈妈是一个出纳,经常需要做一些统计报表的工作.今天是妈妈的生日,小Q希望可以帮妈妈分担一些工作,作为她的生日礼物之一. ...
- bzoj1058: [ZJOI2007]报表统计
set.操作:insert(u,v)在u后面插入v,若u后面已插入过,在插入过的后面插入.mingap求出序列两两之间差值的最小值.minsortgap求出排序后的序列两两之间的最小值.用multis ...
随机推荐
- Flask从入门到精通之Flask表单渲染成HTML
表单字段是可调用的,在模板中调用后会渲染成HTML.假设视图函数把一个NameForm 实例通过参数form 传入模板,在模板中可以生成一个简单的表单,如下所示: <form method=&q ...
- 模仿 AppStore 顶部动画
App Store 顶部动画 App Store 中 Games.Apps.Updates 的顶部动画的特点: 自然状态下是大标题,右边有一个 button 顶上去时,变成小标题,右边按钮消失 导航栏 ...
- Android访问网络,使用HttpURLConnection还是HttpClient?
本文转自:http://blog.csdn.net/guolin_blog/article/details/12452307,感谢这位网友的分享,谢谢. 最近在研究Volley框架的源码,发现它在HT ...
- pythonweb框架Flask学习笔记01-ubuntu18.04下安装Flask
Flask 依赖两个外部库: Jinja2 模板引擎和 Werkzeug WSGI 工具集 由于各个项目工程之间可能存在python库版本的差异 为了防止库版本差异对项目开发产生的影响,使用virtu ...
- h5预订酒店项目|html5酒店模板|h5酒店webapp开发
近几天尝试着使用html5+css3+swiper+jqUI+layerMobile等技术开发了一款仿携程.去哪儿.艺龙webapp酒店预订系统,页面图标统一使用iconfont,仿原生app右侧弹窗 ...
- 线性表 (单链表、循环链表-python实现)
一.线性表 线性表的定义: 线性表是具有相同数据类型的有限数据的序列. 线性表的特点: 出了第一个元素外,每个元素有且仅有一个直接前驱,除最后一个元素外有且只有一个后继. 线性表是一种逻辑结构,表示元 ...
- Eclipse调用hadoop2运行MR程序(转)
hadoop:hadoop2.2 ,windows myeclipse环境: Eclipse调用hadoop运行MR程序其实就是普通的java程序可以提交MR任务到集群执行而已.在Hadoop1中,只 ...
- [转] 使用 MVC 5 的 EF6 Code First 入门 系列
译文:http://www.cnblogs.com/Bce-/category/573301.html 原文:http://www.asp.net/mvc/overview/getting-start ...
- python:rs, ws, es = select.select(inputs, [], []) --报错error 10022
昨晚折腾的1个多钟,直到3点多才睡,感觉自己也是热爱代码了,敲3个多钟一点也不累(其实是为了凌晨6点起来抢票回家了^_^) 练习python中select进行异步通信-防止socket.recv方法阻 ...
- android学习-IPC机制之ACtivity绑定Service通信
bindService获得Service的binder对象对服务进行操作 Binder通信过程类似于TCP/IP服务连接过程binder四大架构Server(服务器),Client(客户端),Serv ...