勤快的love枫[ZJOI2007]
题目描述
小绝恋love 枫是一个出纳,经常需要做一些统计报表的工作。今天是绝恋love 枫的生日,小绝恋love 枫希望可以帮爸爸分担一些工作,作为他的生日礼物之一。经过仔细观察,小绝恋love 枫发现统计一张报表实际上是维护一个非负整数数列,并且进行一些查询操作。在最开始的时候,有一个长度为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。于是小绝恋love 枫写了一个程序,使得程序可以自动完成这些操作,但是他发现对于一些大的报表他的程序运行得很慢,你能帮助他改进程序么?
输入
第一行包含两个整数N,M,分别表示原数列的长度以及操作的次数。
第二行为N 个整数,为初始序列。
接下来的M 行每行一个操作,即“INSERT i k”,“MIN_GAP”,“MIN_SORT_GAP”中的一种(无多余空格或者空行)。
输出
对于每一个“MIN_GAP”和“MIN_SORT_GAP”命令,输出一行答案即可。
样例输入
3 5
5 3 1
INSERT 2 9
MIN_SORT_GAP
INSERT 2 6
MIN_GAP
MIN_SORT_GAP
样例输出
2
2
1
提示
对于30% 的数据,N≤1000,M≤5000
对于100% 的数据,N,M≤50000
对于所有的数据,序列内的整数不超过5*10^8。
题解
一道显而易见的数据结构题,在数据结构题里几乎算不用动脑子的那种。前后最小的一下子想到钢哥讲过的垃圾堆,维护两个堆就可以实现有效答案的添加和删除。原数列也可以用一个副本数组,只维护目前最靠前和最靠后的值(考试时忘了下一个位置的开头元素还需要记录,炸了不少分)。但是整个序列中最相近的两个元素,就有些犯难。明明知道就应该建个平衡树找前驱后继,尴尬的情况是平衡树只打过Treap,Treap只打过两道题,还是一两周之前的事了,考场上打挂可能性极大。权衡了一下,还是拿了个数组暴力排序,唯一的优化是如果最小值已经为0就不再改动。算上这道题平衡树也只打过三道,简直不能算做过题。书到用时方恨少,事非经过不知难,从第一次见到这句话到现在已经有些年了,理解倒是越来越深了= =。
#include<iostream>
#include<cstdio>
#include<cstring>
#include<queue>
#include<cstdlib>
#include<cmath>
using namespace std;
const int sj=;
int n,m,a[sj],a1,a2,mi,g,q,h,jq,hj,size,d[sj];
priority_queue<int,vector<int>,greater<int> > qi;
priority_queue<int,vector<int>,greater<int> > du;
char c[];
struct tree
{
int l,r,v,rnd;
}t[sj];
int bj(int x,int y)
{
return x<y?x:y;
}
void lturn(int &x)
{
int tt=t[x].r;
t[x].r=t[tt].l;
t[tt].l=x;
x=tt;
}
void rturn(int &x)
{
int tt=t[x].l;
t[x].l=t[tt].r;
t[tt].r=x;
x=tt;
}
void query_pre(int k,int x)
{
if(k==) return;
if(t[k].v<x)
{
q=k;
query_pre(t[k].r,x);
}
else query_pre(t[k].l,x);
}
void query_beh(int k,int x)
{
if(k==) return;
if(t[k].v>x)
{
h=k;
query_beh(t[k].l,x);
}
else query_beh(t[k].r,x);
}
void cr(int &x,int y)
{
if(x==)
{
size++;
x=size;
t[x].rnd=rand();
t[x].v=y;
return;
}
if(t[x].v==y)
{
jq=hj=y;
q=h=;
return;
}
if(y>t[x].v)
{
cr(t[x].r,y);
if(t[x].rnd>t[t[x].r].rnd) lturn(x);
}
if(y<t[x].v)
{
cr(t[x].l,y);
if(t[x].rnd>t[t[x].l].rnd) rturn(x);
}
}
void init()
{
scanf("%d%d",&n,&m);
mi=0x7fffffff;
for(int i=;i<=n;i++)
{
scanf("%d",&a[i]);
if(i!=) qi.push(abs(a[i]-a[i-]));
if(mi)
{
q=h=-;
jq=hj=;
cr(g,a[i]);
if(q==-)
{
query_beh(g,a[i]);
query_pre(g,a[i]);
if(q!=-||h!=-)
{
jq=(q==-)?0x7fffffff:t[q].v;
hj=(h==-)?0x7fffffff:t[h].v;
mi=bj(mi,abs(a[i]-jq));
mi=bj(mi,abs(a[i]-hj));
}
}
else mi=;
}
memcpy(d,a,sizeof(a));
}
}
void cl()
{
for(int i=;i<=m;i++)
{
scanf("%s",c);
if(c[]=='R')
{
scanf("%d%d",&a1,&a2);
if(a1!=n)
{
qi.push(abs(a2-a[a1+]));
du.push(abs(a[a1+]-d[a1]));
}
qi.push(abs(a2-d[a1]));
d[a1]=a2;
if(mi)
{
q=h=-;
jq=hj=;
cr(g,a2);
if(q==-)
{
query_beh(g,a2);
query_pre(g,a2);
if(q!=-||h!=-)
{
jq=(q==-)?0x7fffffff:t[q].v;
hj=(h==-)?0x7fffffff:t[h].v;
mi=bj(mi,abs(a2-jq));
mi=bj(mi,abs(a2-hj));
}
}
else mi=;
}
}
if(c[]=='G')
{
while(!qi.empty()&&!du.empty()&&qi.top()==du.top())
{
qi.pop();
du.pop();
}
printf("%d\n",qi.top());
}
if(c[]=='S') printf("%d\n",mi);
}
}
int main()
{
init();
cl();
return ;
}
love
在cogs上看到了这题,交了一下RE,发现数据范围是十倍……调了数组大小再交一遍,居然TLE了!据说堆可能会炸掉,不得已又改成了线段树。跑得确实快了,可是比堆难写得多啊。差点上两百行【抵制恶意缩行不良风气,从我做起!
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<cmath>
using namespace std;
const int sj=;
int n,m,a[sj],a1,a2,mi,g,q,h,jq,hj,size,d[sj],last;
struct xs
{
int v,l,r;
}xds[sj*];
char c[];
struct tree
{
int l,r,v,rnd;
}t[sj];
int bj(int x,int y)
{
return x<y?x:y;
}
void lturn(int &x)
{
int tt=t[x].r;
t[x].r=t[tt].l;
t[tt].l=x;
x=tt;
}
void rturn(int &x)
{
int tt=t[x].l;
t[x].l=t[tt].r;
t[tt].r=x;
x=tt;
}
void query_pre(int k,int x)
{
if(k==) return;
if(t[k].v<x)
{
q=k;
query_pre(t[k].r,x);
}
else query_pre(t[k].l,x);
}
void query_beh(int k,int x)
{
if(k==) return;
if(t[k].v>x)
{
h=k;
query_beh(t[k].l,x);
}
else query_beh(t[k].r,x);
}
void cr(int &x,int y)
{
if(x==)
{
size++;
x=size;
t[x].rnd=rand();
t[x].v=y;
return;
}
if(t[x].v==y)
{
jq=hj=y;
q=h=;
return;
}
if(y>t[x].v)
{
cr(t[x].r,y);
if(t[x].rnd>t[t[x].r].rnd) lturn(x);
}
if(y<t[x].v)
{
cr(t[x].l,y);
if(t[x].rnd>t[t[x].l].rnd) rturn(x);
}
}
void build(int x,int z,int y)
{
xds[x].l=z;
xds[x].r=y;
xds[x].v=0x7fffffff;
if(z==y) return;
int mid=(z+y)>>;
build(x<<,z,mid);
build((x<<)|,mid+,y);
}
void update(int x,int z,int y)
{
if(xds[x].l==xds[x].r&&xds[x].l==z)
{
xds[x].v=y;
return;
}
int mid=(xds[x].l+xds[x].r)>>;
if(z<=mid)
{
update(x<<,z,y);
xds[x].v=bj(xds[(x<<)|].v,xds[x<<].v);
}
else
{
update((x<<)|,z,y);
xds[x].v=bj(xds[(x<<)|].v,xds[x<<].v);
}
}
void init()
{
scanf("%d%d",&n,&m);
mi=0x7fffffff;
build(,,sj*-);
for(int i=;i<=n;i++)
{
scanf("%d",&a[i]);
if(i!=) update(,i-,abs(a[i]-a[i-]));
if(mi)
{
q=h=-;
jq=hj=;
cr(g,a[i]);
if(q==-)
{
query_beh(g,a[i]);
query_pre(g,a[i]);
if(q!=-||h!=-)
{
jq=(q==-)?0x7fffffff:t[q].v;
hj=(h==-)?0x7fffffff:t[h].v;
mi=bj(mi,abs(a[i]-jq));
mi=bj(mi,abs(a[i]-hj));
}
}
else mi=;
}
}
memcpy(d,a,sizeof(a));
last=n-;
}
void cl()
{
for(int i=;i<=m;i++)
{
scanf("%s",c);
if(c[]=='R')
{
scanf("%d%d",&a1,&a2);
last++;
update(,last,abs(a2-d[a1]));
update(,a1,abs(a2-a[a1+]));
d[a1]=a2;
if(mi)
{
q=h=-;
jq=hj=;
cr(g,a2);
if(q==-)
{
query_beh(g,a2);
query_pre(g,a2);
if(q!=-||h!=-)
{
jq=(q==-)?0x7fffffff:t[q].v;
hj=(h==-)?0x7fffffff:t[h].v;
mi=bj(mi,abs(a2-jq));
mi=bj(mi,abs(a2-hj));
}
}
else mi=;
}
}
if(c[]=='G') printf("%d\n",xds[].v);
if(c[]=='S') printf("%d\n",mi);
}
}
int main()
{
init();
cl();
return ;
}
form
勤快的love枫[ZJOI2007]的更多相关文章
- bzoj-1096 1096: [ZJOI2007]仓库建设(斜率优化dp)
题目链接: 1096: [ZJOI2007]仓库建设 Description L公司有N个工厂,由高到底分布在一座山上.如图所示,工厂1在山顶,工厂N在山脚.由于这座山处于高原内陆地区(干燥少雨),L ...
- BZOJ 1096: [ZJOI2007]仓库建设 [斜率优化DP]
1096: [ZJOI2007]仓库建设 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 4201 Solved: 1851[Submit][Stat ...
- 【BZOJ1060】[ZJOI2007]时态同步 树形DP
[BZOJ1060][ZJOI2007]时态同步 Description 小Q在电子工艺实习课上学习焊接电路板.一块电路板由若干个元件组成,我们不妨称之为节点,并将其用数字1,2,3-.进行标号.电路 ...
- BZOJ 1093 [ZJOI2007] 最大半连通子图(强联通缩点+DP)
题目大意 题目是图片形式的,就简要说下题意算了 一个有向图 G=(V, E) 称为半连通的(Semi-Connected),如果满足图中任意两点 u v,存在一条从 u 到 v 的路径或者从 v 到 ...
- BZOJ4011: [HNOI2015]落忆枫音
Description 「恒逸,你相信灵魂的存在吗?」 郭恒逸和姚枫茜漫步在枫音乡的街道上.望着漫天飞舞的红枫,枫茜突然问出 这样一个问题. 「相信吧.不然我们是什么,一团肉吗?要不是有灵魂……我们 ...
- 【BZOJ】【1059】【ZJOI2007】矩阵游戏
二分图完美匹配/匈牙利算法 如果a[i][j]为黑点,我们就连边 i->j ,然后跑二分图最大匹配,看是否有完美匹配. <_<我们先考虑行变换:对于第 i 行,如果它第 j 位是黑点 ...
- 【BZOJ】【4011】【HNOI2015】落忆枫音
拓扑排序+DP 题解:http://blog.csdn.net/PoPoQQQ/article/details/45194103 http://www.cnblogs.com/mmlz/p/44487 ...
- bzoj1058: [ZJOI2007]报表统计
set.操作:insert(u,v)在u后面插入v,若u后面已插入过,在插入过的后面插入.mingap求出序列两两之间差值的最小值.minsortgap求出排序后的序列两两之间的最小值.用multis ...
- 洛谷 P1169 [ZJOI2007]棋盘制作
2016-05-31 14:56:17 题目链接: 洛谷 P1169 [ZJOI2007]棋盘制作 题目大意: 给定一块矩形,求出满足棋盘式黑白间隔的最大矩形大小和最大正方形大小 解法: 神犇王知昆的 ...
随机推荐
- v9更新栏目缓存提示PHP has encountered a Stack overflow解决方法
原因: 客户在把一些栏目删除或者新增栏目时没更新栏目缓存导致v9_category表里有原来的垃圾信息,多余的表. 解决方法:通过phpmyadmin找到栏目表出错的条目,修改错误信息. 具体步骤: ...
- Angular中使用Swiper不能滑动的解决方法
Swiper是目前较为流行的移动端触摸滑动插件,因为其简单好用易上手,很受很多设计师的欢迎. 今天在使用Swiper的时候遇到这个问题: 使用angularjs动态循环生成swiper-slide类, ...
- win7开启telnet客户端
- 如何相互转换逗号分隔的字符串和List
将逗号分隔的字符串转换为List 方法 1: 利用JDK的Arrays类 [java] view plain copy ico_fork.svg1.5 KB String str = " ...
- mybatis 查询 xml list参数
mybatis 查询 xml list参数: <select id="getByIds" resultType="string" parameterTyp ...
- 我的第一篇博文:C++最初的路-经典的小游戏走迷宫
写在开始:这个博客建于大二下学期.2年多的学习,从网上借鉴的大牛经验,代码,指导数不胜数,而其中大部分来自别人的博客,于是期待有一天也能把自己在学习过程中的一些经验拿出来与大家分享. 其实我凝望了C+ ...
- 浅谈angular中的promise
promise目的就是为了跳出回调地狱.老掉牙的东西,大神轻拍. 举个最简单的例子:请求数据(getData),解析数据(executeData),显示数据(showData). //获取数据 fun ...
- nodejs模块学习: connect2解析
nodejs模块学习: connect2 解析 nodejs 发展很快,从 npm 上面的包托管数量就可以看出来.不过从另一方面来看,也是反映了 nodejs 的基础不稳固,需要开发者创造大量的轮子来 ...
- Unity3D拖尾组件在Ui界面下正常显示
在项目中Canvas下UI添加拖尾效果,会发现Ui完全遮挡住了拖尾. 如果要正常显示通常需要对Canvas进行设置,Render Mode 我这里用的是-Camera模式 其次要对Material 下 ...
- SpringMvc支持跨域访问,Spring跨域访问,SpringMvc @CrossOrigin 跨域
SpringMvc支持跨域访问,Spring跨域访问,SpringMvc @CrossOrigin 跨域 >>>>>>>>>>>> ...