【BZOJ4864】【BJWC2017】神秘物质 - Splay
题意:
Description
21ZZ 年,冬。小诚退休以后, 不知为何重新燃起了对物理学的兴趣。 他从研究所借了些实验仪器,整天研究各种微观粒子。这一天, 小诚刚从研究所得到了一块奇异的陨石样本, 便迫不及待地开始观测。 在精密仪器的视野下,构成陨石的每个原子都无比清晰。 小诚发现, 这些原子排成若干列, 每一列的结构具有高度相似性。于是,他决定对单独一列原子进行测量和测试。被选中的这列共有 N 个顺序排列的原子。 最初, 第 i 个原子具有能量 Ei。 随着时间推移和人为测试, 这列原子在观测上会产生两种变化:
merge x e 当前第 x 个原子和第 x+1 个原子合并,得到能量为 e 的新原子;
insert x e 在当前第 x 个原子和第 x+1 个原子之间插入一个能量为 e 的新原子。
对于一列原子,小诚关心的是相邻一段中能量最大和能量最小的两个原子的能量差值,
称为区间极差。 因此, 除了观测变化外,小诚还要经常统计这列原子的两类数据:
max x y 当前第 x 到第 y 个原子之间的任意子区间中区间极差的最大值;
min x y 当前第 x 到第 y 个原子之间的任意子区间中区间极差的最小值。
其中, 子区间指的是长度至少是 2 的子区间。
小诚坚信这项研究可以获得诺贝尔物理学奖。为了让小诚早日了结心愿,你能否帮助他实现上述的观测和测量呢?
Input
Output
输出若干行, 按顺序依次表示每次 max 和 min 类事件的测量结果。
题解:
看到merge和insert操作肯定是splay维护啦……
考虑如何处理询问,极差最大值肯定是区间最大值减最小值,这个不用说;
极差最小值呢?
结论是极差最小值一定是两个相邻的数之差;
考虑三个数,肯定选中间的数和与其差较小的另一个数极差最小,三个数都选不会优于这个选择,类似归纳即可;
那么splay维护子树最大/最小值,前驱后继和子树内最小极差,pushup的时候搞搞即可。
代码:
#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<queue>
#define inf 1000000007
#define eps 1e-9
using namespace std;
typedef long long ll;
typedef double db;
struct node{
int son[],fa,v,l,r,siz,mi,mx,mm;
}t[];
int n,m,x,y,rt,cnt,num[];
char op[];
bool Son(int u){
return t[t[u].fa].son[]==u;
}
void pushup(int u){
t[u].l=t[u].son[]?t[t[u].son[]].l:u;
t[u].r=t[u].son[]?t[t[u].son[]].r:u;
t[u].siz=t[t[u].son[]].siz+t[t[u].son[]].siz+;
t[u].mx=max(t[u].v,max(t[t[u].son[]].mx,t[t[u].son[]].mx));
t[u].mi=min(t[u].v,min(t[t[u].son[]].mi,t[t[u].son[]].mi));
t[u].mm=min(min(t[t[u].son[]].mm,t[t[u].son[]].mm),min(t[u].son[]?abs(t[u].v-t[t[t[u].son[]].r].v):inf,t[u].son[]?abs(t[u].v-t[t[t[u].son[]].l].v):inf));
}
int build(int l,int r,int ff){
int mid=(l+r)/;
t[mid].fa=ff;
if(l<mid)t[mid].son[]=build(l,mid-,mid);
if(mid<r)t[mid].son[]=build(mid+,r,mid);
pushup(mid);
return mid;
}
void rotate(int u){
int f=t[u].fa,ff=t[f].fa,ch=Son(u),cf=Son(f);
t[f].son[ch]=t[u].son[ch^];
t[t[f].son[ch]].fa=f;
t[ff].son[cf]=u;
t[u].son[ch^]=f;
t[u].fa=ff;
t[f].fa=u;
pushup(f);
pushup(u);
}
void splay(int u,int to){
for(;t[u].fa!=to;rotate(u)){
int f=t[u].fa;
if(t[f].fa!=to)rotate(Son(u)^Son(f)?u:f);
}
if(!to)rt=u;
}
int findx(int u,int k){
int nw=u;
for(;;){
if(t[t[nw].son[]].siz+==k)return nw;
if(k<=t[t[nw].son[]].siz){
nw=t[nw].son[];
}else{
k-=t[t[nw].son[]].siz+;
nw=t[nw].son[];
}
}
}
int split(int x,int y){
int l=findx(rt,x-),r=findx(rt,y+);
//printf("%d %d\n",l,r);
splay(l,);
splay(r,rt);
return t[t[rt].son[]].son[];
}
void merge(int x,int y){
int u=split(x+,x+);
t[u].v=y;
t[u].son[]=t[u].son[]=;
pushup(u);
pushup(t[rt].son[]);
pushup(rt);
}
void ins(int x,int y){
int u=split(x+,x+);
t[t[rt].son[]].son[]=++cnt;
t[cnt].fa=t[rt].son[];
t[cnt].v=y;
pushup(cnt);
pushup(t[rt].son[]);
pushup(rt);
}
int getmx(int x,int y){
int u=split(x+,y+);
return t[u].mx-t[u].mi;
}
int getmi(int x,int y){
int u=split(x+,y+);
return t[u].mm;
}
int main(){
scanf("%d%d",&n,&m);
cnt=n+;
for(int i=;i<=n;i++){
scanf("%d",&t[i+].v);
}
t[].mx=;
t[].mi=t[].mm=inf;
cnt=n+;
rt=build(,n+,);
for(int i=;i<=m;i++){
scanf("%s%d%d",op,&x,&y);
if(op[]=='e'){
merge(x,y);
}else if(op[]=='n'){
ins(x,y);
}else if(op[]=='a'){
printf("%d\n",getmx(x,y));
}else{
printf("%d\n",getmi(x,y));
}
}
return ;
}
【BZOJ4864】【BJWC2017】神秘物质 - Splay的更多相关文章
- 【BZOJ4864】神秘物质 [Splay]
神秘物质 Time Limit: 10 Sec Memory Limit: 256 MB Description Input Output Sample Input Sample Output 1 ...
- [BZOJ4864][BeiJing2017Wc]神秘物质(splay)
首先merge就是先delete两次再insert,Max就是整个区间的最大值减最小值,Min就是区间中所有相邻两数差的最小值. Splay支持区间最大值,区间最小值,区间相邻差最小值即可. #inc ...
- BZOJ 4864 [BJWC2017]神秘物质 (splay)
题目大意: 让你维护一个序列,支持: 1.合并两个相邻的数为一个新的数 2.在某个位置插入一个数 3.查询一个区间的任意子区间极差的最大值 4.查询一个区间的任意子区间极差的最小值 前两个操作可以用$ ...
- 【BZOJ4864】[BeiJing 2017 Wc]神秘物质 Splay
[BZOJ4864][BeiJing 2017 Wc]神秘物质 Description 21ZZ 年,冬. 小诚退休以后, 不知为何重新燃起了对物理学的兴趣. 他从研究所借了些实验仪器,整天研究各种微 ...
- [bzoj4864][BeiJing2017Wc]神秘物质_非旋转Treap
神秘物质 bzoj-4864 BeiJing-2017-Wc 题目大意:给定一个长度为n的序列,支持插入,将相邻两个元素合并并在该位置生成一个指定权值的元素:查询:区间内的任意一段子区间的最大值减最小 ...
- BZOJ4864: [BeiJing 2017 Wc]神秘物质(Splay)
Description 21ZZ 年,冬. 小诚退休以后, 不知为何重新燃起了对物理学的兴趣. 他从研究所借了些实验仪器,整天研究各种微观粒子.这 一天, 小诚刚从研究所得到了一块奇异的陨石样本, 便 ...
- 【bzoj4864】神秘物质
Description 给出一个长度为n的序列,第i个数为ai,进行以下四种操作: merge x e:将当前第x个数和第x+1个数合并,得到一个新的数e: insert x e:在当前第x个数和第x ...
- BZOJ_4864_[BeiJing 2017 Wc]神秘物质_Splay
BZOJ4864_[BeiJing 2017 Wc]神秘物质_Splay Description 21ZZ 年,冬. 小诚退休以后, 不知为何重新燃起了对物理学的兴趣. 他从研究所借了些实验仪器,整天 ...
- BZOJ 4864: [BeiJing 2017 Wc]神秘物质 解题报告
4864: [BeiJing 2017 Wc]神秘物质 Description 21ZZ 年,冬. 小诚退休以后, 不知为何重新燃起了对物理学的兴趣. 他从研究所借了些实验仪器,整天研究各种微观粒子. ...
随机推荐
- 听说”双11”是这么解决线上bug的
听说"双11"是这么解决线上bug的 --Android线上热修复的使用与原理 预备知识和开发环境 Android NDK编程 AndFix浅析 Android线上热修复的原理大同 ...
- Delicious Apples (hdu 5303 贪心+枚举)
Delicious Apples Time Limit: 5000/3000 MS (Java/Others) Memory Limit: 524288/524288 K (Java/Other ...
- Leetcode45:Intersection of Two Linked Lists
Write a program to find the node at which the intersection of two singly linked lists begins. For ex ...
- eclipse配置默认的jdk
1.window-preferences-java-Installed JREs-Add-Standard VM-[添加jre6或jre7的路径]
- SRM 621 D2L3: MixingColors, math
题目:http://community.topcoder.com/stat? c=problem_statement&pm=10409&rd=15854 利用高斯消元求线性空间的基,也 ...
- instanceof运算符的使用
在之前的学习中,经常遇到instanceof运算符,对于它的用法总感觉理解不到位,所以专门总结一下它的用法加深理解. instanceof主要用来判断一个类是否实现了某个接口,或者判断一个实例对象是否 ...
- 使用RabbitMQ放置自己定义对象(不借助序列化工具,比如protobuffer)V2.0
怎样使用RabbitMQ盛放自己定义的对象呢?一般都会使用序列化工具在投放之前转换一次.从MQ取回的时候再逆序列化还原为本地对象.这里使用C++自带的强制类型装换,将本地对象的内存模型当做自然的序列化 ...
- devenv.exe 编译Solution
Build https://docs.microsoft.com/en-us/visualstudio/ide/reference/build-devenv-exe Builds a solution ...
- poj--1664--放苹果(递归好体)
放苹果 Time Limit: 1000MS Memory Limit: 10000KB 64bit IO Format: %I64d & %I64u Submit Status De ...
- thinkphp实现多数据库操作
这篇文章主要介绍了ThinkPHP实现多数据库连接的解决方法,需要的朋友可以参考下 ThinkPHP实现连接多个数据的时候,如果数据库在同一个服务器里的话只需要这样定义模型: ? 1 2 3 cl ...