bzoj3000 Big Number 数论,斯特林公式

Posted Kaiser

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了bzoj3000 Big Number 数论,斯特林公式相关的知识,希望对你有一定的参考价值。

Description

给你两个整数N和K,要求你输出N!的K进制的位数。

Input

有多组输入数据,每组输入数据各一行,每行两个数——N,K

Output

每行一个数为输出结果

Sample Input

2 5
2 10
10 10
100 200

Sample Output

1
1
7
69
对于100%的数据,有2≤N≤2^31, 2≤K≤200,数据组数T≤200。

题解

用Stirling公式求近似值

位数=logk(n!)+1

≈ logk(sqrt(2πn)*(n/e)^n)+1

= logk( sqrt(2πn))+log[(n/e)^n]+1

=1/2*logk( 2πn)+nlog(n/e)+1

=0.5*logk ( 2πn)+nlog(n/e)+1

=0.5*logk ( 2πn)+nlog(n)-nlog(e)+1

PS:pi=acos(-1.0),e=exp(1)

PS2:eps的存在是为了防止n=2,k=2这样刚好的情况出现,这个时候向上取整要多取1位

 

斯特林公式是求解n!的近似解,对于较大的n数值是十分准确的。

所以可以通过数学方法解决。

 1 #include<cstring>
 2 #include<cmath>
 3 #include<iostream>
 4 #include<cstdio>
 5 #include<algorithm>
 6 
 7 #define ll long long
 8 using namespace std;
 9 const double eps=0.00000000001;
10 const double pai=3.14159265359;
11 const double e=exp(1);
12 
13 int n,k;
14 
15 int main()
16 {
17     freopen("fzy.in","r",stdin);
18     freopen("fzy.out","w",stdout);
19     while(~scanf("%d%d",&n,&k))
20     {
21         if (n<=1000)
22         {
23             double ans=0;
24             for (int i=1;i<=n;i++)
25                 ans+=log(i);
26             ans/=log(k);
27             int res=ceil(ans+eps);
28             printf("%d\\n",res);
29         }
30         else
31         {
32             double res=(log(sqrt(2*pai*n))+n*log(n/e))/log(k);
33             ll ans=ceil(res-eps);
34             printf("%lld\\n",ans);
35         }
36     }
37 }

对了,c++小数处理的时候会有精度损失的问题,所以需要适当加上一个小数

 

以上是关于bzoj3000 Big Number 数论,斯特林公式的主要内容,如果未能解决你的问题,请参考以下文章

1018 Big Number 简单数论

斯特林公式-Stirling公式(取N阶乘近似值)-HDU1018-Big Number 牛客网NowCoder 2018年全国多校算法寒假训练营练习比赛(第三场)A.不凡的夫夫

bzoj 4026 dC Loves Number Theory (主席树+数论+欧拉函数)

bzoj4671 异或图(斯特林反演,线性基)

BZOJ4555求和(第二类斯特林数,组合数学,NTT)

BZOJ1130 N的阶乘的长度 V2(斯特林近似)