hdu 1018

hdu 1018

http://acm.hdu.edu.cn/showproblem.php?pid=1018

求N的阶乘的位数,1<=N<=10^7

解题方法:

针对每个非负整数n,计算其n!的位数,由于n的位数很大,我们不可能通过直接计算得到结果 
1.设a=log10(n!) ,则n!=10^a,其中a是一个小数
2.设a=x+y,其中 x为整数,y为小数
3.因此 n!=10^x+10^y
4.10^x肯定为10的倍数,决定了n!的位数,10^y为(1~10,不取10),决定n!的各位数字
5.因此,只要知道了a就可以求出n!的位数
6.因为a= log10(n!)=log10(n)+ log10(n-1)+……log10(2)+log10(1),所以a的值可以很容易求出 

另解:求N的阶乘的为数其结果相当于求小于N的最大素数,因此可以用求素数或素数筛法的方法求(不超时的情况下),当然也能用斯特林公式,可以当做斯特林公式的另一种应用了吧【其实是当时题目理解错误,还以为自己发现了什么不得了的东西哈哈】

相关知识:

斯特林公式简介:斯特林公式(Stirling's approximation)是一条用来取n的阶乘近似值的数学公式。一般来说,当n很大的时候,n阶乘的计算量十分大,所以斯特林公式十分好用,而且,即使在n很小的时候,斯特林公式的取值已经十分准确。

表达式: n!≈√(2πn)·(n/e)^n

利用斯特林(Stirling)公式的进行求解。下面是推导得到的公式:
res=(long)( (log10(sqrt(4.0*acos(0.0)*n)) + n*(log10(n)-log10(exp(1.0)))) + 1 );
n=1的时候,上面的公式不适用,所以要单独处理n=1的情况!

#include<iostream>
#include<cmath>
using namespace std;
int main()
{
    int n,test,i,ans;
    double t;
   cin>>test;
    while(test--)
    {
       cin>>n;
       t=0;
        for(i=2;i<=n;i++)
           t+=log10(i*1.0); 
        ans=int(t)+1;
       cout<<ans<<endl;
    }
    return 0;
}






  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值