博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
N!分解素因子及若干问题
阅读量:4577 次
发布时间:2019-06-08

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

 

将N!表示成

N! = p1^t1*p2^t2*…pi^ti…*pk^tk(其中p1,p2……pk是素数,1<N<= 10^6)

显然很容易通过素数筛选求出pi,因为1<pi<=N,关键是如何快速地求出ti。

我们先来看一下对于2这个素因子,把N!分成两部分,即奇偶两部分

 

假设N是偶数

N!

=1*2*3*4*5……N

=(2*4*6……) * (1*3*5……)

因为有N/2个偶数,所以偶数部分可以提出N/2个2,

=2^(N/2) * (1*2*3*……N/2) * (1*3*5*……)

=2^(N/2) * (N/2)! * (1*3*5*……)

 

看到了吗!神奇的事情发生了,N规模的问题转化成了N/2的问题了。上面假设了N是偶数,当然N是奇数时也是一样的,只要规定这里的除法是取整就可以了

 

于是有递推公式 f(n,2) = f(n/2,2) + n/2,表示n!中2的个数。

 

用同样的方法可以推出 f(n,p) = f(n/p,p) + n/p,表示n!中素数p的个数。

 

于是有代码

 

int f(int n,int p){    if(n==0) return 0;    return f(n/p) + n/p;}

 

将问题推广一下:

 

问题1:N!的末尾有几个0?

因为 10 = 2*5,所以只要知道N!有多少个2和多少个5,问题就解决了。Min(f(n,2),f(n,5)) 显然f(n,2)>f(n,5),所以问题就转化成了求f(n,5)。

 

问题2:N!的转化成12进制之后,末尾有几个0?

和问题一样,12=2*2*3,所以只要求Min(f(n,2)/2,f(n,3)),就可以了。

 

问题3: 求组合数C(n,m)(mod p)

C(n,m)=n!/(m!*(n-m)!) ,只要对分子和分母分别分解素因子,然后因为C(n,m)肯定是整数,所以C(n,m)肯定可以表示成p1^t1*p2^t2*......pi^ti的形式,只要拿分子素因子的幂减去分母对应的素因子的幂即可。好了,后面就简单了,二分快速幂取模......

转载于:https://www.cnblogs.com/AbandonZHANG/archive/2012/07/18/2598328.html

你可能感兴趣的文章
JS实现——贪吃蛇
查看>>
推荐10款免费的在线UI测试工具
查看>>
解构控制反转(IoC)和依赖注入(DI)
查看>>
燕十八redis 微博地址
查看>>
面向对象的特征有哪些方面?
查看>>
三月十一号
查看>>
关于java类加载器的一些概念
查看>>
JNI.ZC_文件(.so/.h)位置
查看>>
JAVA基础——数据流
查看>>
线性代数之——克拉默法则、逆矩阵和体积
查看>>
OpenCV_累加一个三通道矩阵中的所有元素
查看>>
20162308 2016-2017-2 《程序设计与数据结构》第1周学习总结
查看>>
hdu 6057 Kanade's convolution(子集卷积)
查看>>
1126 求递推序列的第N项(51nod)
查看>>
用纯css改变默认的radio和checkbox的样式
查看>>
Linux的cron和crontab
查看>>
fcitx 无法启动
查看>>
图片旋转---老外写的代码
查看>>
PHP中过滤html标签(转)
查看>>
bzoj 2288: 【POJ Challenge】生日礼物【链表+堆】
查看>>