20200409 Running Median

题意翻译

输入 个数,每次到奇数个数的时候输出中位数。

多测, 组数据。


题解

使用对顶堆。

一个小根堆,存放比中位数大的数。

一个大根堆,存放比中位数小的数。

动态调整堆的大小。

参考了神仙的 std 写法,细节处理比我之前的写法好的多。


priority_queue 的坑点

  • priority_queue < int, vector < int >, greater < int > > 可以直接定义小根堆

  • priority_queue 的成员函数 size 返回类型为 unsigned int 型,如果直接在 if 的条件中调用 size 函数并进行运算,一定要写 (int)q.size()


code

#include<bits/stdc++.h>
using namespace std;

typedef long long LL;

template < typename Tp >
void read(Tp &x) {
    x = 0; int fh = 1; char ch = 1;
    while(ch != '-' && (ch < '0' || ch > '9')) ch = getchar();
    if(ch == '-') fh = -1, ch = getchar();
    while(ch >= '0' && ch <= '9') x = x * 10 + ch - '0', ch = getchar();
    x *= fh;
}

int T, n;

void Init(void) {
    read(T);
}

priority_queue <int> q1;
priority_queue <int, vector < int >, greater < int > > q2;

void solve(void){

    while(q1.size()) q1.pop();
    while(q2.size()) q2.pop();

    int cas;
    read(cas);read(n);

    printf("%d %d\n", cas, (n + 1) / 2);

    read(cas);
    q1.push(cas);
    printf("%d ", cas);
    for(int i = 2, x; i <= n; i++) {
        read(x);
        if(x <= q1.top()) q1.push(x);
        else q2.push(x);
    //    printf("\n %d - 1  ** q1 = %d, q2 = %d \n", i, q1.size(), q2.size());
        if((int)q1.size() - (int)q2.size() > 1) {
            q2.push(q1.top()); q1.pop();
        }
        else if((int)q2.size() - (int)q1.size() > 1) {
            q1.push(q2.top()); q2.pop();
        }

//        printf("\n %d - 2  ** q1 = %d, q2 = %d \n", i, q1.size(), q2.size());

        if(i % 2 == 1) {
            if((int)q1.size() > (int)q2.size()) printf("%d ", q1.top());
            else printf("%d ", q2.top());
            if(i % 20 == 19 && i != n) puts("");
        }
    }
    puts("");
}

void Work(void) {
    while(T--) {
        solve();
    }
}

int main(void) {
    Init();
    Work();
    return 0;
}