博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
HDU-3092 Least common multiple---数论+分组背包
阅读量:6445 次
发布时间:2019-06-23

本文共 1708 字,大约阅读时间需要 5 分钟。

题目链接:

题目大意:

有一个数字n,现在要把它分解成几个数字相加!然后这几个数字有最小公倍数,题目目的是求出最大的最小公倍数。我们知道所有的素数或者其指数方相加可以表示其它的数字,而把n分解之后求其公倍数自然是互质的数字直接相乘最大,所以目的就变成了求n能分解之后由素数或者其指数数,只要他们之间相互互质就行。

解题思路:

将n分解成不同素数之和,这样就是两两互质,求出的lcm是最大的,而且不只是素数之和,可以分解成素数的k次方,这样不同的素数的k次方之间仍然是互质的。

这样变成了分组背包,每一个素数就是一组,这一组中只能选一个,而背包容量为S。求出最大的价值即可。

还有一个问题,由于价值过大,需要取模,但是随便取模的话就不好判断大小,所以采用取对数的方法,一个数组记录取对数的值,一个数据记录答案,每次按照取对数的数组的大小判断是否需要更新,需要更新的话直接更新该数组和答案数组。

1 #include
2 using namespace std; 3 typedef long long ll; 4 ll n, m; 5 const int maxn = 3000 + 10; 6 bool is_prime[maxn]; 7 ll prime[maxn], tot, ans[maxn]; 8 double dp[maxn]; 9 void init(int n)10 {11 for(int i = 2; i <= n; i++)is_prime[i] = 1;12 for(int i = 2; i <= n; i++)13 if(is_prime[i])14 {15 prime[tot++] = i;16 for(int j = i * 2; j <= n; j += i)is_prime[j] = 0;17 }18 //for(int i = 0; i < tot; i++)cout<
<
> n >> m)24 {25 for(int i = 0; i <= n; i++)dp[i] = 0, ans[i] = 1;26 for(int i = 0; i < tot && prime[i] <= n; i++)27 {28 double tmp = log(prime[i]);29 for(int j = n; j >= 1; j--)30 {31 ll k = prime[i], cnt = 1;32 while(k <= j)33 {34 if(dp[j] < dp[j - k] + tmp * cnt)35 {36 dp[j] = dp[j - k] + tmp * cnt;37 ans[j] = ans[j - k] * k % m;38 }39 cnt++;40 k *= prime[i];41 }42 }43 //for(int i = 1; i <= n; i++)cout<
<

 

转载于:https://www.cnblogs.com/fzl194/p/9033277.html

你可能感兴趣的文章
多线程问题及处理方法【转】
查看>>
Sublime for Python
查看>>
Ubuntu10.04.4 Server下基于LVS DR模式+Keepalived的负载均衡高可用
查看>>
《Activiti实战》摘抄&笔记1
查看>>
ladon 与 suds互通 (转)
查看>>
Java8新特性(三)
查看>>
合成(Composite)模式
查看>>
Win7安装MySQL Server 5.6记录
查看>>
syscat.tables系统表的npages和fpages区别
查看>>
js 小数取整的函数
查看>>
java 错误: 找不到或无法加载主类 HelloWorld.class
查看>>
ssh 登录云服务器
查看>>
安全连接Nexus自有仓库
查看>>
在Debian6.0(Squeeze)安装Memcached 和PHP5-Memcached
查看>>
神州数码人才测评【03】数学(原创解析)
查看>>
AJAX POST请求中参数以form data和request payload形式在servlet中的获取方式
查看>>
Python 内层名字空间访问外层名字空间中的变量
查看>>
使用spark操作ensemble
查看>>
Android内核开发:系统分区与镜像文件的烧写
查看>>
【Java多线程】写入同一文件,自定义线程池与线程回收利用
查看>>