【BZOJ4524】[Cqoi2016]伪光滑数 堆(模拟搜索)

【BZOJ4524】[Cqoi2016]伪光滑数

Description

若一个大于1的整数M的质因数分解有k项,其最大的质因子为Ak,并且满足Ak^K<=N,Ak<128,我们就称整数M为N-伪光滑数。现在给出N,求所有整数中,第K大的N-伪光滑数。

Input

只有一行,为用空格隔开的整数N和K

2 ≤ N ≤ 10^18, 1 ≤ K ≤ 800000,保证至少有 K 个满足要求的数

Output

只有一行,为一个整数,表示答案。

Sample Input

12345 20

Sample Output

9167

题解:先打表打出前31个质数,然后枚举每个质数作为最大质因子,算出此时最多能有多少项,那么此时的最大值显然=这个质数^项数。我们将其扔到堆中。

然后我们要做的就是每次从堆中取出最大的数,将他的一个质因子变小,然后再扔到队列中去,重复k次。并且我们要保证我们的取法不会出现遗漏和重复。这就是一个套路了。我们试图模拟搜索的过程。在搜索时,假如我们已经取了一些质数,他们的积为val,那么我们可以再取一个val的最小质因子,或者继续考虑下一个更小的质因子。现在我们要模拟这个方法:

我们维护四元组(val,mn,first,second)代表当前的数,最小的质因子编号,最小的质因子次数,次小的质因子次数。那么如果我们从堆中取出一个数,我们可以用一个最小质因子来换一个次小质因子,或者用最小质因子的下一个质数来替换最小质因子。容易发现这样是可以做到不重不漏的。由于每次我们只往堆中扔进去2个数,所以时间复杂度就是$O(Klog_K)$的。(看见那些时间复杂度是$O(31*Klog_K)$的我就想笑~)

#include <cstdio>
#include <cstring>
#include <iostream>
#include <queue>
#include <algorithm>
using namespace std;
typedef long long ll;
int m;
ll n;
int p[]={1,2,3,5,7,11,13,17,19,23,29,31,37,41,43,47,53,59,61,67,71,73,79,83,89,97,101,103,107,109,113,127};//31
struct node
{
	ll val;
	int mn,lst,sec;
	bool operator < (const node &a) const
	{
		return val<a.val;
	}
};
priority_queue<node> q;
int main()
{
	scanf("%lld%d",&n,&m);
	register node x,y;
	register int i,j;
	for(i=1;i<=31;i++)
	{
		ll tmp;
		for(tmp=1,j=0;tmp<=n/p[i];j++,tmp*=p[i]);
		x.val=tmp,x.mn=i,x.lst=j-1,x.sec=0;
		q.push(x);
	}
	while(--m)
	{
		x=q.top(),q.pop();
		if(x.mn)
		{
			y.val=x.val/p[x.mn]*p[x.mn-1],y.mn=x.mn-1,y.lst=1,y.sec=x.lst-1;
			q.push(y);
		}
		if(x.sec)
		{
			y.val=x.val/p[x.mn+1]*p[x.mn],y.mn=x.mn,y.lst=x.lst+1,y.sec=x.sec-1;
			q.push(y);
		}
	}
	x=q.top();
	printf("%lld",x.val);
	return 0;
}
时间: 2024-07-31 07:35:42

【BZOJ4524】[Cqoi2016]伪光滑数 堆(模拟搜索)的相关文章

【BZOJ-4524】伪光滑数 堆 + 贪心 (暴力) [可持久化可并堆 + DP]

4524: [Cqoi2016]伪光滑数 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 183  Solved: 82[Submit][Status][Discuss] Description 若一个大于R的整数J的质因数分解有F项,其最大的质因子为ak,并且满足ak^k≤N, ak<128,我们就称整数J为N-伪光滑数. 现在给出L,求所有整数中,第E大的N-伪光滑数. Input 只有一行,为用空格隔开的整数L和E. 2 ≤ N ≤ 10^1

@bzoj - [email&#160;protected] [Cqoi2016]伪光滑数

目录 @description@ @[email protected] @version - [email protected] @version - [email protected] @accepted [email protected] @[email protected] @description@ 若一个大于 1 的整数 M 的质因数分解有 k 项,其最大的质因子为 \(A_k\),并且满足 \(A_k^k \le N\),\(A_k < 128\),我们就称整数 M 为 N - 伪光

[CQOI2016]伪光滑数

题目描述 若一个大于1的整数M的质因数分解有k项,其最大的质因子为Ak,并且满足Ak^K<=N,Ak<128,我们就称整数M为N-伪 光滑数.现在给出N,求所有整数中,第K大的N-伪光滑数. 题解 题面的k意思是将这个数质因数分解后所有的质因子的指数和. 我们先把128以内的所有素数找出来,然后做一个dp. 我们令dp[i][j]表示当前数的最大的质因子为p[i],当前所有素因子的指数和为j的数的集合. 我们再令g[i][j]表示当指数和位j时,所有最大质因子小于等于p[i]的数的集合. 然后

bzoj4524【CQOI2016】伪光滑数

4524: [Cqoi2016]伪光滑数 Time Limit: 10 Sec  Memory Limit: 512 MB Submit: 257  Solved: 117 [Submit][Status][Discuss] Description 若一个大于R的整数J的质因数分解有F项,其最大的质因子为ak,并且满足ak^k≤N, ak<128,我们就称整数J为N-伪光滑数. 现在给出L,求所有整数中,第E大的N-伪光滑数. Input 只有一行,为用空格隔开的整数L和E. 2 ≤ N ≤ 1

【CQOI 2016】伪光滑数

Solution 又是一道神仙题.蒟蒻表示不看题解根本不会做 首先我们定义一个 DP 数组 \(\mathtt{f[i][j]}\) 表示:最大质因子为 \(\mathtt{p[i]}\),分解成 j 个质数(可以相同)组成的集合(其中 \(\mathtt{f[i][j]}\) 是这个集合的根节点,在这里我们用左偏树).我们知道,只要得到了这个 DP,我们就可以把这个玩意儿的权值塞进队列排序,再插入儿子进行查询. 为了求得这个数组,我们需要再定义一个 \(\mathtt{g[i][j]}\) 表

CSS伪类选择器active模拟JavaScript点击事件

一.说明 设置元素在被用户激活(在鼠标点击与释放之间发生的事件)时的样式. IE7及更早浏览器只支持a元素的:active,从IE8开始支持其它元素的:active. 另:如果需要给超链接定义:访问前,鼠标悬停,当前被点击,已访问这4种伪类效果,而又没有按照一致的书写顺序,不同的浏览器可能会有不同的表现.超链接的4种状态,需要有特定的书写顺序才能生效.注意,a:hover必须位于a:link和a:visited之后,a:active必须位于a:hover之后.可靠的顺序是:l(link)ov(v

poj 3087 Shuffle&#39;m Up (模拟搜索)

Shuffle'm Up Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 5953   Accepted: 2796 Description A common pastime for poker players at a poker table is to shuffle stacks of chips. Shuffling chips is performed by starting with two stacks of

模拟 --- 搜索模拟

Robot Motion Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 10130   Accepted: 4932 Description A robot has been programmed to follow the instructions in its path. Instructions for the next direction the robot is to move are laid down in

种树 (堆模拟网络流)

种树 题目描述 cyrcyr今天在种树,他在一条直线上挖了n个坑.这n个坑都可以种树,但为了保证每一棵树都有充足的养料,cyrcyr不会在相邻的两个坑中种树.而且由于cyrcyr的树种不够,他至多会种k棵树.假设cyrcyr有某种神能力,能预知自己在某个坑种树的获利会是多少(可能为负),请你帮助他计算出他的最大获利. 输入输出格式 输入格式: 第一行,两个正整数n,k. 第二行,n个正整数,第i个数表示在直线上从左往右数第i个坑种树的获利. 输出格式: 输出1个数,表示cyrcyr种树的最大获利