这道题是一道splay裸题,然而身为蒟蒻的我并不会,所以这道题我维护的是一个大根堆与一个小根堆结合起来的类似沙漏的结构。

本题难点在于询问的不是最大最小值,而是第K小值,所以我们想到了维护这样两个堆,上面是一个大小限定为K-1的大根堆,下面是一个小根堆,每次插入/查询操作时,保持前K-1大的始终在大根堆内。

插入/查询函数:

int heap[200005][3],hsize[3];
int m,n,num[200005],temp;
void put(int x,int i){
heap[++hsize[i]][i]=x;
int son=hsize[i],pa=son>>1;
while(pa>=1){
if(i==1){
if(heap[pa][i]>heap[son][i]){
int t=heap[pa][i];heap[pa][i]=heap[son][i];heap[son][i]=t;
son=pa;
pa>>=1;
}else break;
}else {
if(heap[pa][i]<heap[son][i]){
int t=heap[pa][i];heap[pa][i]=heap[son][i];heap[son][i]=t;
son=pa;
pa>>=1;
}else break;
}
}
}
int get(int i){
int rv=heap[1][i];
heap[1][i]=heap[hsize[i]--][i];
int pa=1,son;
while(pa<=(hsize[i]>>1)){
if(i==1) son=heap[pa<<1][i]<heap[pa<<1|1][i]?pa<<1:pa<<1|1;
else son=heap[pa<<1][i]>heap[pa<<1|1][i]?pa<<1:pa<<1|1;
if(i==1){
if(heap[pa][i]>heap[son][i]){
int t=heap[pa][i];heap[pa][i]=heap[son][i];heap[son][i]=t;
pa=son;
}else break;
}else {
if(heap[pa][i]<heap[son][i]){
int t=heap[pa][i];heap[pa][i]=heap[son][i];heap[son][i]=t;
pa=son;
}else break;
}
}
return rv;
}

对整体的插入/查询:

void putt(int x){
put(x,1);
if(x==heap[1][1]){
get(1);
put(x,2);
int t=get(2);
put(t,1);
}
}
int gett(){
temp++;
int rv=get(1);
put(rv,2);
return rv;
}

附:跑得飞快的代码:

#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstdlib>
#include <cstring>
#include <cmath>
using namespace std;
int read(){
int rv=0,fh=1;
char c=getchar();
while(c<'0'||c>'9'){
if(c=='-') fh=-1;
c=getchar();
}
while(c>='0'&&c<='9'){
rv=(rv<<1)+(rv<<3)+c-'0';
c=getchar();
}
return rv*fh;
}
int heap[200005][3],hsize[3];
int m,n,num[200005],temp;
void put(int x,int i){
heap[++hsize[i]][i]=x;
int son=hsize[i],pa=son>>1;
while(pa>=1){
if(i==1){
if(heap[pa][i]>heap[son][i]){
int t=heap[pa][i];heap[pa][i]=heap[son][i];heap[son][i]=t;
son=pa;
pa>>=1;
}else break;
}else {
if(heap[pa][i]<heap[son][i]){
int t=heap[pa][i];heap[pa][i]=heap[son][i];heap[son][i]=t;
son=pa;
pa>>=1;
}else break;
}
}
}
int get(int i){
int rv=heap[1][i];
heap[1][i]=heap[hsize[i]--][i];
int pa=1,son;
while(pa<=(hsize[i]>>1)){
if(i==1) son=heap[pa<<1][i]<heap[pa<<1|1][i]?pa<<1:pa<<1|1;
else son=heap[pa<<1][i]>heap[pa<<1|1][i]?pa<<1:pa<<1|1;
if(i==1){
if(heap[pa][i]>heap[son][i]){
int t=heap[pa][i];heap[pa][i]=heap[son][i];heap[son][i]=t;
pa=son;
}else break;
}else {
if(heap[pa][i]<heap[son][i]){
int t=heap[pa][i];heap[pa][i]=heap[son][i];heap[son][i]=t;
pa=son;
}else break;
}
}
return rv;
}
void putt(int x){
put(x,1);
if(x==heap[1][1]){
get(1);
put(x,2);
int t=get(2);
put(t,1);
}
}
int gett(){
temp++;
int rv=get(1);
put(rv,2);
return rv;
}
int f[200005];
int main(){
freopen("in.txt","r",stdin);
m=read();n=read();
for(int i=1;i<=m;i++){
num[i]=read();
}
for(int i=1;i<=n;i++){
f[read()]++;
}
for(int i=1;i<=m;i++){
putt(num[i]);
while(f[i]){
printf("%d\n",gett());
f[i]--;
}
}
/*for(int i=1;i<=10000;i++){
put(rand(),2);
}
for(int i=1;i<=10000;i++){
printf("%d ",get(2));
}*/
fclose(stdin);
return 0;
}

洛谷 [P1801] 黑匣子的更多相关文章

  1. 洛谷 P1801 黑匣子_NOI导刊2010提高(06)(未完)

    P1801 黑匣子_NOI导刊2010提高(06) 题目描述 Black Box是一种原始的数据库.它可以储存一个整数数组,还有一个特别的变量i.最开始的时候Black Box是空的.而i等于0.这个 ...

  2. 洛谷 - P1801 - 黑匣子 - 对顶堆

    这道题是提高+省选-的难度,做出来的话对数据结构题目的理解会增加很多. 可以使用一种叫做对顶堆的东西,对顶堆是在线维护第n小的logn的算法.大概的思路是,假如我们要找的是第n小,我们就维护一个大小为 ...

  3. 洛谷P1801 黑匣子

    题目传送门 分析:这题和另外一个题目中位数非常相似,有兴趣可以先看看,比这一题简单.首先暴力模拟还是别想了,估计30%的数据都有点悬.正解应该是用二叉堆.但是如果用一个堆当然不方便,所以建两个堆,一个 ...

  4. [洛谷P1801]黑匣子_NOI导刊2010提高(06)

    题目大意:两个操作:向一个可重集中加入一个元素:询问第$k$大的数($k$为之前询问的个数加一) 题解:离散化,权值线段树直接查询 卡点:无 C++ Code: #include <cstdio ...

  5. 洛谷 P1801 黑匣子_NOI导刊2010提高(06)

    题目描述 Black Box是一种原始的数据库.它可以储存一个整数数组,还有一个特别的变量i.最开始的时候Black Box是空的.而i等于0.这个Black Box要处理一串命令. 命令只有两种: ...

  6. 洛谷 P1801 黑匣子_NOI导刊2010提高(06) 题解

    昨晚恶补了一下二叉堆的内容 然后就找了几个二叉堆的题来做awa 然后发现用二叉堆做这题复杂度是O(nlogn) 但是有O(n)的解法 (某大佬这么说) 思路大概就是: 利用一个大根堆一个小根堆来维护第 ...

  7. 洛谷 P1801 黑匣子 题解

    题面 离线处理: 大体思路就是将数组排序,然后对于第k次询问把不可行的数打上标记,然后从头开始寻找第k个没打标记的点的值(排序后的数组保证了它是第k小的). 实现方法:首先离散化原始数组,得到数组fi ...

  8. 【洛谷P1801】黑匣子

    黑匣子 题目链接 看到题解中“维护两个堆”,突然想到了这道题的解法 维护两个堆:大根堆h1, 小根堆h2 大根堆里的是最小的i个值,小根堆里是剩下的值 每Add一个值时 插入到小根堆中, 再比较小根堆 ...

  9. 「洛谷 P1801」黑匣子

    好像很久没有更过博客了,因为博主这几周很忙.其实是在搞颓. 题意很难懂,所以就不重复了.其实是懒. 一眼看上去这是个 \(Splay\) 裸题,直接插入一个数,查询区间第 \(K\) 大,但是这样太不 ...

随机推荐

  1. UWP: 通过命令行启动 UWP 应用

    最近在开发应用的过程中,我遇到了如标题所述的需求,其实主要是为了能够快捷启动应用,正像我们可以在"运行"对话框中可以输入一些可执行程序的名称后,就能够直接启动它:这样做,可以增加 ...

  2. VisualSVN Server的配置和使用方法

    VisualSVN Server是免费的,而VisualSVN是收费的.VisualSVN是SVN的客户端,和Visual Studio集成在一起, VisualSvn Server是SVN的服务器端 ...

  3. 解决无限 This file is indented with tabs instead of 4 spaces

    File -> Settings -> Editor -> Code Style -> Java -> Tabs and Indents -> Use tab ch ...

  4. php表单提交并发送邮件给某个邮箱(示例源码)

    今天老板要求做一个需求,在官网上做一个页面提交的表单,并且当表单点击后,把表单的内容直接提交并通过发送邮件的方式到老板指定的邮箱,下面就分享 一下我的做法 首先建立一个html文档,把页面制作好,并且 ...

  5. Core Graphics框架是Quartz的核心,也是内容描画的基本接口。

    Core Graphics框架是Quartz的核心,也是内容描画的基本接口.

  6. NSString拼接字符串

    NSString* string; // 结果字符串 02 NSString* string1, string2; //已存在的字符串,需要将string1和string2连接起来 03   04 / ...

  7. java开发常用jar包

    mail.jar与activation.jar 里面包含了activation.jar和mail.jar两个包.通过里面的类的调用便可以达到发送电子邮件的目的 commons-beanutils.ja ...

  8. 在windows64位Anaconda3环境下安装XGBoost

    安装步骤参考的是: "Installing XGBoost For Anaconda on Windows":https://www.ibm.com/developerworks/ ...

  9. CSS3 动画及过渡详解

    今天开始我们一起来学习有关于CSS3制作动画的几个属性:变形(transform).过渡(transition)和动画(animation)等CSS3技术. 首先我们先来了解一下变形(transfor ...

  10. JS实现倒计时

    HTML部分: <div class="div"> <div id="div"> </div> </div> C ...