zoukankan      html  css  js  c++  java
  • K优先队列——对顶堆(两种优先队列的组合)

    题目描述
    你需要维护一个队列,支持以下两种操作:
    1.加入一个非负整数x;
    2.取出当前队列中第k大的数字。
    保证进行第二种操作时,队列中至少有k个数字。
    部分数据经过加密,你需要依次处理每个操作才能获得正确的下一个操作。
    输入
    第一行包括三个非负整数n,k,p,分别表示操作次数,参数k以及数据是否进行过加密。
    接下来n行,每行先给出一个数opt,表示操作类型。若opt=1,接下来还会有一个非负整数x,若p=0,表示往队列中加入x,若p=1,表示往队列中加入x异或上前一次出队操作取出的数字后得到的结果,如果还未进行过出队操作,把前一次取出的数字看作0。若opt=2,表示要求取出并输出当前队列中第k大的数字。
    输出
    对于每一个出队操作,输出一个正整数表示答案。
    样例输入

    5 2 1
    1 2
    1 3
    2
    1 3
    2
    

    样例输出

    2
    1
    

    提示
    对于100%的数据,1 ≤ k ≤ n ≤ 2∗105 ,0 ≤ x ≤ 109

    之前在upc也做过关于对顶堆的题,在洛谷 还有这道题

    对顶堆是由大根堆小根堆组成的一种数据结构,经常用来处理在线第 k 大(小),或者是动态中位数的问题
    在这里插上一张图
    图片来源:https://www.luogu.com.cn/user/119261
    在这里插入图片描述
    在这一道题目中适合上面放小根堆,下面放大根堆,小根堆里面的元素的顺序是从小到大排序的,大根堆里面的元素是从大到小排列的。小根堆的堆顶为小根堆中最小的元素,大根堆中的元素是大根堆中最大的元素。

    利用这一特点,用上面的小根堆储存大的 k 个元素,其余的元素储存小的部分元素,这样一来,这两个堆也有单调性,即小根堆的堆顶(小根堆中最小的元素)大于大根堆的堆顶(大根堆中最大的元素),满足从上到下递减的关系,所以,每次小根堆的堆顶就是第 K 大的元素。

    目标就是动态维护小根堆中尽量满足 K 个元素,不足时如果大根堆中还有元素,就将大根堆中的元素放入小根堆(运行错误在这里),意思就是说,如果大根堆中没有元素就没有办法pop()

    code:

    #include <bits/stdc++.h>
    using namespace std;
    #define wuyt main
    typedef long long ll;
    #define HEAP(...) priority_queue<__VA_ARGS__ >
    #define heap(...) priority_queue<__VA_ARGS__,vector<__VA_ARGS__ >,greater<__VA_ARGS__ > >
    template<class T> inline T min(T &x,const T &y){return x>y?y:x;}
    template<class T> inline T max(T &x,const T &y){return x<y?y:x;}
    ll read(){ll c = getchar(),Nig = 1,x = 0;while(!isdigit(c) && c!='-')c = getchar();
        if(c == '-')Nig = -1,c = getchar();
        while(isdigit(c))x = ((x<<1) + (x<<3)) + (c^'0'),c = getchar();
        return Nig*x;}
    #define read read()
    const ll inf = 1e15;
    const ll INF = 0x3f3f3f3f;
    const int maxn = 3e5 + 7;
    const int mod = 998244353;
    #define pi 3.14159265358979323846
    ll res,ans;
    ll n,k,p;
    priority_queue <ll ,vector<ll> ,greater<ll> > xiaogen;
    priority_queue <ll ,vector<ll> ,less<ll> > dagen;
    void in_it(ll x){
        if(xiaogen.size()<k) xiaogen.push(x);
        else{
            if(xiaogen.size()&&xiaogen.top()<x){
                dagen.push(xiaogen.top());
                xiaogen.pop();
                xiaogen.push(x);
            }
            else dagen.push(x);
        }
    }
    int main()
    {
        n=read,k=read,p=read;
        ll temp=0;
        while(n--){
            ll opt=read;
            if(opt==1){
                ll x=read;
                if(p==0) in_it(x);
                else in_it(x^temp);
            }
            else{
                temp = xiaogen.top();
                printf("%lld
    ",temp);
                xiaogen.pop();
                if(dagen.size()){
                    xiaogen.push(dagen.top());
                    dagen.pop();
                }
            }
        }
        return 0;
    }
    /**
    5 2 1
    1 2
    1 3
    2
    1 3
    2
    **/
    
    
  • 相关阅读:
    FileUpload的使用
    关于hibernate4的配置我要好好反省一下
    比较SQL Server 2000 数据库中两个库的差异
    用google生活
    用OWC11图形分析本页面及其他页面Table中的数据
    请教ASP.NET培训应该培训的内容和以及顺序
    最近一个快要结束的项目的BUG分析
    我也发软件开发团队的思考(侧重点是人员)
    一个SQL语句的问题,我百思不得其解,请教各位
    分享C#高端视频教程
  • 原文地址:https://www.cnblogs.com/PushyTao/p/13144139.html
Copyright © 2011-2022 走看看