| Time Limit: 1000MS | Memory Limit: 10000K | |
| Total Submissions: 7099 | Accepted: 2888 |
Description
N Transaction i Black Box contents after transaction Answer
(elements are arranged by non-descending)
1 ADD(3) 0 3
2 GET 1 3 3
3 ADD(1) 1 1, 3
4 GET 2 1, 3 3
5 ADD(-4) 2 -4, 1, 3
6 ADD(2) 2 -4, 1, 2, 3
7 ADD(8) 2 -4, 1, 2, 3, 8
8 ADD(-1000) 2 -1000, -4, 1, 2, 3, 8
9 GET 3 -1000, -4, 1, 2, 3, 8 1
10 GET 4 -1000, -4, 1, 2, 3, 8 2
11 ADD(2) 4 -1000, -4, 1, 2, 2, 3, 8
Input
Output
Sample Input
7 4 3 1 -4 2 8 -1000 2 1 2 6 6
Sample Output
3 3 1 2
Source
题意不好懂。。我直接解释下样例什么意思把。
7 4 3 1 -4 2 8 -1000 2 1 2 6 67代表下面给定7个数的数字序列,4可以理解为四次查询, 1 2 6 6为查讯,第一次查询是求数字序列只有前一个数(3)时,此时的第一小的数字,即1,第二次查询是求数字序列只有前2个数(3 1)时,此时的第二小的数字,即 3,第三次查询是求数字序列只有前6个数时(3,1,-4,2,8,-1000),此时的第三小的数字,即1,第四次查询是求数字序列只有前6个数时,此时的第四小的数字。
思路为: 当求第4小的数字时,我们用一个大顶堆来维护前三个最小的数字,那么小顶堆的堆顶即为所求。
用priority_queue<int>big; 优先队列还起到大顶堆的作用,顶部即为最大值 ,即 big.top();
priority_queue<int,vector<int>,greater<int> >small; 小顶堆,顶堆为最小值 ,即 small.top();
用样例来说明一下 大顶堆和小顶堆工作方法:
1 2 6 6
首先是1: 把第一个数3 插入到小顶堆中,这时候判断,如果大顶堆不为空且小顶堆的top小于大顶堆的top时,就把二者的top值互换,因为,大顶堆中的数不能比小顶堆中的大,大顶堆维护第i次查询时,前i-1个最小的数,这时候大顶堆为空,不用互换值, 输出第一次查询时前1个数的第1小的数即为 小顶堆的top 3,然后把小顶堆的top 移除,放到大顶堆中。
然后是2: 把第二个数1插入到小顶堆中,判断,大顶堆不为空,小顶.top() 1 <大顶.top() 3 ,所以二者互换,小顶.top()为3,大顶top()为1, 然后输出小顶.top(),即为第二小的数。把小顶的.top()移除,放入大顶中。这是大顶中维护的是目前数字中前两个最小的数。
然后是6: 要求前6个数中第3小的数字,先得把3 1以后的四个数字插入才能够六个数,依次插入, -4插入小顶堆,这时 大顶堆 3 1,-4<3,互换, 小顶堆 3,大顶堆为1 -4 ,2插入小顶堆,小顶堆 2,3 大顶堆 1 -4, 2>1,不用互换, 8插入到小顶堆,小顶堆为 2,3,8 大顶堆为 1,-4, 2>1,不用互换,-1000插入到小顶堆,小顶堆 -1000,2,3,8,大顶堆1,-4, -1000<1,不行,互换以后得小顶堆,1,2,3,8 , 大顶堆 -1000,-4 ,输出小顶堆.top() 1, 并把它移除放入到大顶堆中,为下次查询做准备。
所以从以上可以看出,关键点就是第i次查询时,大顶堆中维护的总是目前数字中最小的 i-1个数。
代码:
#include <iostream>
#include <stdio.h>
#include <queue>
#include <algorithm>
using namespace std;
const int maxn=30010;
int num[maxn];
int n,m;
int main()
{
priority_queue<int>big;
priority_queue<int,vector<int>,greater<int> >small;
int m,n;
scanf("%d%d",&m,&n);
for(int i=1;i<=m;i++)
scanf("%d",&num[i]);
int cnt=1;
int op;
for(int i=1;i<=n;i++)
{
cin>>op;
while(cnt<=op)
{
small.push(num[cnt]);
if(!big.empty()&&small.top()<big.top())//小顶堆里面的数不能比大顶堆里面的数小
{
int n1=big.top();
int n2=small.top();
big.pop();
small.pop();
big.push(n2);
small.push(n1);
}
cnt++;
}
printf("%d\n",small.top());
big.push(small.top());//这句话很关键,保证了在求第k个最小数时,大顶堆里面保存的是前k-1个最小数
small.pop();
}
return 0;
}
[ACM] POJ 1442 Black Box (堆,优先队列),布布扣,bubuko.com
[ACM] POJ 1442 Black Box (堆,优先队列)
原文:http://blog.csdn.net/sr_19930829/article/details/37922859