ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

递增全局第k小

2022-04-09 11:00:55  阅读:159  来源: 互联网

标签:lfloor int 递增 每次 times 小根堆 Theta 全局


洛谷P7072

一句话题意:给定 正整数 \(w \in [1,99]\) ,向空队列中不断加数,设当前有 \(n\) 个数,每次加完求第 \(\lfloor n \times w \% \rfloor\) 大的数是多少。 \((1 \le n \le 1e5\))

做法:看起来是求区间第 \(k\) 大,有点复杂,但由于 \(n \times w \%\) 是单调递增的,我们可以用一个奇技淫巧。

用一个大根堆,一个小根堆维护。

每加一个数,加入小根堆,但要控制小根堆的size为当前的 \(\lfloor n \times w \% \rfloor\) ,所以如果当前size大了,就把多的丢掉。

但是这样会产生一个问题:我们现在丢掉的有可能是以后的答案,所以我们要拿个 \(rubbish\) 数组存一下丢掉的东西,每次先倒回来,算完答案再倒进去。

但是这样每次放过去放过来 \(n(1 - w\%)\) 个数,放n次,会使复杂度达到 \(\Theta(n(1 - w\%) * n) =\) \(\Theta(n^2)\) ,不彳亍!

但我们发现,我们每次的 \(\lfloor n \times w\% \rfloor\) 最多只会增加1,所以每次只有可能有一位是有问题的。而这一位必定是 \(rubbish\) 中最大的一个,所以只用一个大根堆维护最大值,每次往回吐一个即可。至此,一次复杂度变成 \(\Theta(logn)\) 了!

#include<bits/stdc++.h>
using namespace std;
const int N = 1e5 + 10;
int n, w, x;
priority_queue <int,vector<int>,greater<int> > xgd;
priority_queue <int> dgd;
int main()
{
	scanf("%d%d", &n, &w);
	int now;
	for(int i = 1; i <= n; ++i){
		scanf("%d", &x);
		xgd.push(x);
		now = max(1, i * w / 100);
		if(dgd.size()) xgd.push(dgd.top()), dgd.pop();
		while(xgd.size() > now) dgd.push(xgd.top()), xgd.pop();
		cout << xgd.top() <<' ';
	}
	return 0;
}

标签:lfloor,int,递增,每次,times,小根堆,Theta,全局
来源: https://www.cnblogs.com/fakeryu/p/16121155.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有