题目:http://www.lydsy.com/JudgeOnline/problem.php?id=1053

感想:这道题拿到以后还是想去知道一个数的约数个数要怎么求,去网上搜了公式,但是还是没有思路,最后看了好几个大佬的博客我才勉强知道这道题怎么做

  其实我看见这数据范围我就下意识觉得搜索会爆,但是实际上这道题不会爆的

方法:首先约数的个数求法

  已知数n=2^a + 3^b + 5^c + 7^d +……+ 第i个素数^x

  约数个数=(a+1)*(b+1)*(c+1)*(d+1)*……*(x+1)

  在知道这个公式后就可以用暴搜了,但是还是要知道就是数据范围内,素数最多就12个,所以预处理前几个素数

  还是看代码吧,我在代码中加了一些注释

 #include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<cstdlib>
#include<cmath>
#define maxn 2000000005
using namespace std; const int prime[]={,,,,,,,,,,,,,,,}; int n,m,ans,ans_tot; //设一个数n=2^a + 3^b + 5^c + 7^d +……+ prime[dep]^x;
//则这个数的约数个数为(a+1)*(b+1)*(c+1)*(d+1)*……*(x+1)
//这是约数个数计算公式 void fuck(int dep,long long now_val,int tot,int num)
{
//dep=当前是prime中的素数的下标
//now_val是当前的值
//tot指当前的约数的个数
//num是上一个素数(prime[dep-1])的幂
if(dep==)
{
if(now_val>ans&&tot>ans_tot){
ans=now_val;ans_tot=tot;
}//当发现比ans还大的数且这个数的约数个数比ans多
//即找到一个更优的解
if(now_val<=ans&&tot>=ans_tot)
{
ans=now_val;ans_tot=tot;
} //当发现一个比ans小但是约数比ans多的数
//说明ans不是一个反素数,也可以更新答案
return; }
int t=;
for(int i=;i<=num;i++)
{
fuck(dep+,now_val*t,tot*(i+),i);
//搜下一个素数,又i从0次方开始的,所以now_val*s,s=1;
//约数的公式可以知道这是tot*(i+1)
//到这个时候,prime[dep]已经i次方了,所以幂是i
//其实可以想到最优解是小素数的幂越大越优,所以加num限定
t*=prime[dep];//t表示当前素数prime[dep]的i次方
if(now_val*t>n)break;//当乘不下去了就跳出 } } int main()
{
scanf("%d",&n);
fuck(,,,);
printf("%d",ans);
}

以上皆为看了大佬博客后的理解,如有不对,请大佬们指出

04-27 16:38