icodelab 最多的约数
描述
对于一个正整数a,如果所有小于a的数的约数个数都小于a本身的约数个数,我认为这个数正是我们所要的。
输入
输入一个正整数X。
输出
输出一个不大于X的且满足上述要求的最大的数a。
输入样例 1
1000
输出样例 1
840
提示
对于10%的数据,1<=n<=1,000 。对于40%的数据,1<=n<=1,000,000。对于100%的数据,1<=n<=2,000,000,000。
代码(为什么没写思路呢?借用老师的代码,注释详细):
#include <cstdio>
#include <iostream>
using namespace std;
typedef long long ll;
int prime[12] = { 0, 2, 3, 5, 7, 11, 13, 17, 19, 23, 29};
// 2*3*5*7*11*13*17*19*23*29>n,所以只需考虑到29即可
ll n, BestSum, BestNum;
//当前走到num这个数,接着用第k个素数,num的约数个数为sum,
//第k个素数的个数上限为limit
void solve(ll num, ll sum, ll limit, ll k) {
if (sum > BestSum) {//找到了约数更多的数,自然保存约数更多的数
BestSum = sum;//保存个数
BestNum = num;//保存数字本身
} else if (sum == BestSum && num < BestNum) { //约数个数一样时,取小数
BestNum = num;
}
for (int i = 1; i <= limit; i++) { //素数k取i个
num *= prime[k];
if (num > n)//num已经超过n了,显然不需要再找下去了
return;
//第2个参数 sum*(1+i) 表示当前num的约数个数,参考质数分解定理
//第3个是参数是i,表示后面的数字的幂不可能超过前面数字的幂
solve(num, sum * (1 + i), i, k + 1);
}
}
int main() {
cin >> n;
//solve(num, sum, limit, k)
//当前走到num这个数,num的约数个数为sum,接着用第k个素数,第k个素数的个数上限为limit
solve(1, 1, 30, 1);
cout << BestNum;
return 0;
}

浙公网安备 33010602011771号