昨晚恶补了一下二叉堆的内容
然后就找了几个二叉堆的题来做awa
然后发现用二叉堆做这题复杂度是O(nlogn)
但是有O(n)的解法
(某大佬这么说)
思路大概就是:
利用一个大根堆一个小根堆来维护第k小,并没有强制在线
不强制在线,所以我们直接读入所有元素,枚举询问,因为
要询问第k小,所以把前面的第k个元素都放进大根堆里面,
然后如果元素数量大于k,就把堆顶弹掉放到小根堆里面,
使大根堆的元素严格等于k,这样这次询问的结果就是小根
堆的堆顶了(前面k-1小的元素都在大根堆里面了)
记得在完成这次询问后重新把小根堆的堆顶放到大根堆里面就好
详见代码:
#include <cstdio> #include <vector> #include <cstring> #include <queue> #define ll long long #define inf 1<<30 #define il inline #define in1(a) read(a) il int max(int x,int y) { return x>y?x:y; } il int min(int x,int y) { return x<y?x:y; } il int abs(int x) { return x>0?x:-x; } il void swap(int &x,int &y) { int t=x; x=y,y=t; } il void readl(ll &x) { x=0; ll f=1; char c=getchar(); while(c<'0'||c>'9') { if(c=='-')f=-f; c=getchar(); } while(c>='0'&&c<='9') { x=x*10+c-'0'; c=getchar(); } x*=f; } il void read(int &x) { x=0; int f=1; char c=getchar(); while(c<'0'||c>'9') { if(c=='-')f=-f; c=getchar(); } while(c>='0'&&c<='9') { x=x*10+c-'0'; c=getchar(); } x*=f; } //上边大部分并没有用 using namespace std; #define N 200010 priority_queue<int,vector<int>,greater<int> > q; priority_queue<int> q1; int n,m,a[N],b[N]; int main() { in1(n),in1(m); for(int i=1; i<=n; i++)in1(a[i]); for(int i=1; i<=m; i++)in1(b[i]); int i=1; for(int j=1; j<=m; j++) { for(; i<=b[j]; i++) { q1.push(a[i]); if(q1.size()==j)q.push(q1.top()),q1.pop(); } printf("%d\n",q.top()); q1.push(q.top()); q.pop(); } return 0; }