高精度 - 阶乘

2017-04-16  本文已影响0人  Veahow

1 题目描述

输入某一个自然数N,求N(0≤N≤10000)的阶乘即N!是多少。

1.1 输入

输入只有一个数N。

1.2 输出

输出N!的结果。

1.3 样例输入

3

1.4 样例输出

6

2 解题

2.1 数学基础

对于某一正整数N,肯定比高其一位的最小整数小,大于甚至等于同位数的最小整数,因此N满足以下关系:

10^{x-1}\leq N<10^x

其中x代表10的幂次,同时也有另外一层含义,即在10的x次方里有x个零。如果再加上最高位的1,x+1就代表了10的x次方的位数。同理,对于10的x-1次方,其位数为x-1+1=x位,x即为正整数N的位数。对以上不等式关系我们取10的对数可得:

\lg10^{x-1}\leq\lg N<\lg10^x

即:

x-1\leq \lg N<x

如果我们定义[k]所表示的为不超过k这个数的最大整数,便可得到:

x-1=[lgN]

我们在等式两边同时加1即可得到N与其位数x的关系

x=[lgN]+1

转化为C++代码中的表示即为int x = (int)log10(N) + 1,特别地,使用log10()函数需要引入头文件<cmath>

2.2 解决思路

题目中很明显指出了输入数N的取值范围,最大为10000。要让计算机求解10000阶乘级别的数值并且全部输出,这对精度和运算速度要求是十分苛刻的,采用常规的递归和循环解法绝对不行。一般来说,进行高精度的计算求解,必须要用到数组来存储各个位的数字,此外要得到精确数值的话可以使用模拟手工运算的算法。但模拟手工运算的算法是有时间瓶颈的,我们可以对此进行优化,不再使用十进制的进位方式,可以采用千进制、万进制的进位方式来减少多余的进位运算。经过测试,十万进制效果显著,因此代码里采用十万进制进位。
我们首先要确定最大数的阶乘即10000!的位数能够达到多少,经过编程计算,10000!的位数为35660,所以我们可以定义一个长度超过35660的数组来存储结果。由于阶乘的运算是递减型的,即10000!=10000×9999×...×1,只要在算法里采用“乘完就进位”的思想,那么对数组里每个元素来说,其值最大绝对不超过10000×10000,这对int类型的变量来说不会发生溢出,因此我们定义的数组类型为整型。
由于输入N的不同,位数也会随之变化,因此需要再定义一个double型数组来存储对数,以便后续的快速查找来计算得出位数。由对数知识:

lg(N!)=lgN+lg(N-1)+...+lg1

可以采用循环来计算对数的值。


3 代码

#include <cstdio>
#include <cmath>

const int MAXN = 10000;
int N;
int ans[4*MAXN];
double lgn[10005];
//定义全局变量默认初始化为0。

int main()
{
    scanf("%d",&N);

    for(int i = 1; i <= 10000; ++i)
        lgn[i] = lgn[i-1] + log10(i);
    //lgn数组存储了0阶乘到10000阶乘的位数,以便快速查找

    ans[0]=1;  //数组最低位必须为1,以便后续进位
    for(int i = 1; i <= N; ++i){
        int x = (int)lgn[i] + 1;

        for(int j = 0, carry_value = 0; j <= x / 5; ++j){
                ans[j] = ans[j] * i + carry_value;
                carry_value = ans[j] / 100000;
                ans[j] %= 100000;
            }
    }
    /*
    采用100000进制的进位,对于100000进制的数,最大为99999,
    因此数组里每个元素都能表示五位的数字,即x/5。在第二层循环
    中为了减少多余的计算,每次循环乘数,循环次数和当前数值位数
    正相关。
    */

    int x = (int)lgn[N] + 1;  //N阶乘的位数
    for(x = x / 5; x >= 0; --x)
        if(ans[x] != 0) break;
    //剔除掉最高位数组元素为0的情况
    printf("%d",ans[x]);  //因为最高位数组元素表示的数字可能不足五位,所以直接输出

    for(int i = x - 1; i >= 0; --i)
        printf("%05d",ans[i]);
    /*
    在输出完最高位数组元素后,其后的元素如果不足五位代表不足
    那几位的数字为0,使用%05d输出格式,输出五位不足五位补0到
    五位
    */
    printf("\n");

    return 0;
}

4 相似题目

上一篇 下一篇

猜你喜欢

热点阅读