题意

$\DeclareMathOperator{\lcm}{lcm}$选 $k$ ($k\le 10$) 个 $1$ 到 $n$($n\le 10^9$)之间的整数(可以相同),使得 $\lcm(a_1, \dots, a_k)$ 最大。

题解

这是 hihoCoder 挑战赛 #6 的 B 题,CLJ(WJMZBMR) 出的。CLJ 的题解:

首先我们注意到,如果你选择了两个不互质的 $a, b$,那么不妨把 $a$ 换成 $a/(a,b)$。显然 LCM 还是不变的。

这意味着存在一组最优解使得所有选择的数都两两互质。

那么我们不妨使用暴搜,首先我们注意到我们至少可以选择比 $n$ 小的最大的 $k$ 个质数来当做初始解。

然后我们从大到小枚举是否使用,搜到 $x$ 时,假如当前最优解是 $ans$, 当前 LCM 是 $w$, 如果还能选择 $t$ 个, 假如 $wx^t \le ans$,那么显然已经无法得出更优的解了,就可以剪枝了。

首先需要指出,上面题解中

如果你选择了两个不互质的数 $a, b$,那么不妨把 $a$ 换成 $a/(a,b)$ 。显然 LCM 还是不变的。

这个结论是错误的,很容易举出反例:$a=4, b=2$,可能是作者笔误。不过,对于不互质的两个数 $a,b$ ,确实存在两个互质的数 $a',b'$ ($a'\le a, b'\le b$),使得 $\lcm(a',b') = \lcm(a,b)$。

写出 $a, b$ 两数的质因子展开式,设

$$

\begin{align}

a = p_1^{i_1} p_2^{i_2}\dots p_n^{i_n} \notag\

b = p_1^{j_1} p_2^{j_2}\dots p_n^{j_n} \notag

\end{align}

$$

在 $a$,$b$ 展开式中,只保留幂次较大的项便得到了 $a'$, $b'$。

至此,我遇到了一个困难:如何求比 $n$($n\le 10^9$)小的最大的 $k$($k\le 10$)个素数?

当然,求出 $k$ 个最大的素数并非我们的最终目的,这样做只是为了得到一个较大的初始解,求出不满 $k$ 个最大的素数也无妨,从而我们可以暴力判断后若干(比如 100)个数。另外,应当能看出初始解是否取一个较大的值,对程序运行时间影响并不大,将其取为 $n$ 甚至 $0$ 也可以。

复杂度

??从递归深度开始考虑??(大误,递归深度最大即为 $k$ 啊!!!我真是沙茶)

Implementation

#include <bits/stdc++.h>
using namespace std; vector<int> a;
long long res;
long double product;
const int mod = 1e9 + 7; int n, k; void dfs(int x, long double cur_prod){
if(a.size() == k || x == 1){
// product = cur_rod;
res = 1;
product = cur_prod;
for(auto i: a){
// product *= i;
res *= i, res %= mod;
}
return;
}
if(cur_prod * pow((long double)x, k - a.size()) <= product)
return; // 剪枝
bool flag = true;
for(auto i: a)
if(__gcd(x, i) != 1){
flag = false;
break;
}
if(flag){
a.push_back(x);
dfs(x-1, cur_prod * x);
a.pop_back();
}
dfs(x-1, cur_prod);
} int main(){
// int n, k;
cin >> n >> k;
// product = n == 1 ? n : n * (n - 1);
product = n;
res = n;
dfs(n, 1);
cout << res << endl;
return 0;
}

上面代码中的 dfs() 还有一种写法:

void dfs(int x, long double cur_prod){
if(a.size() == k || x == 1){
res = 1;
product = cur_prod;
for(auto i: a){
res *= i, res %= mod;
}
return;
} for(int i = x; ; i--){
bool flag = true;
for(auto j: a){
if(__gcd(i, j) != 1){
flag = false;
break;
}
}
if(!flag) continue; if(cur_prod * pow((long double)i, k - a.size()) <= product)
break;
a.push_back(i);
dfs(i, cur_prod * i);
a.pop_back(); }
}

最新文章

  1. 45 个非常有用的 Oracle 查询语句
  2. 高德地图JavaScript开发
  3. zoj The 12th Zhejiang Provincial Collegiate Programming Contest Demacia of the Ancients
  4. HDU 1890:Robotic Sort(Splay)
  5. HDFS入门详解
  6. C#程序中:如何向xml文件中写入数据和读取数据
  7. ng-repeat 遍历同值数组导致的报错
  8. 《Python爬虫学习系列教程》学习笔记
  9. CodeForces 670E Correct Bracket Sequence Editor
  10. 记录一下html渲染页面的 js框架
  11. Redis数据类型SortedSET
  12. PAT B1013
  13. js/jquery遇到的坑总结
  14. SAP MM PO 中的Delivery Date并非保存在EKPO表里
  15. python 之 列表与字典
  16. springboot中velocity tool中注入bean
  17. Linux(centos7)上安装最新版R3.4.1
  18. 【转载】linux 测试机器端口连通性方法
  19. mipi 调试经验【转】
  20. layer获取iframe内容

热门文章

  1. 关于一些Spring MVC控制器的参数注解总结
  2. codeforce Gym 100500H ICPC Quest (简单dp)
  3. CF Gym 100637F The Pool for Lucky Ones
  4. Spark Job具体的物理执行
  5. 跑edgebox
  6. 7.Props向子组件传递数据
  7. c++作业:输入两个整数,用函数求两数之和。函数外部声明有什么作用?
  8. cocos2dx for lua A*寻路算法实现2
  9. python--以1-31的数字作为结尾的列表?论英文好的重要性!
  10. python中的内建函数