【hdoj_1042】N!(大数)

题目:http://acm.hdu.edu.cn/showproblem.php?pid=1042

题目说明待求阶乘的数最大为10000,而10000!的位数为35660(这个数是上网查的),所以已经有的数据类型无法表示.


思路:用int型数组存储n!的每一步计算结果,并且数组大小应该不小于35660这个数.每一步计算,手动模拟乘法的过程.


首先看看乘法的过程.例如,1234 x 67,手动计算,有两种方法.

方法一:先用7乘以1234,再用60乘以1234,然后再将两个结果相加.

方法二:先后用67乘以4、3、2、1.

方法二更容易用代码实现.具体看看方法二的过程,如下图(运算过程在右边)


上述过程可以概括为:先乘当前位,再加上进位数,取余替换前位的值,取商作为新的进位数.


C++代码如下:

#include<iostream>
using namespace std;

int main()
{
	int n;
	const int maxn = 35660 + 1;//10000!的阶乘一共有35660位
	while(cin>>n)
	{
		int a[maxn];
		memset(a,0,sizeof(a));
		a[maxn-1] = 1;//按照从左到右的习惯,记录结果

		int i,j;
		for(i=2;i<=n;i++)//具体计算过程
		{
			int c = 0;
			for(j=maxn-1;j>=0;j--)//用i乘以[1*2*...*(i-1)]的值
			{
				int x = a[j]*i + c;//[这三行是乘法的核心代码,和加法类似]
				a[j] = x % 10;
				c = x / 10;
			}
		}

		for(i=0;i<maxn;i++)//规范输出,找到第一个非零的数的下标
			if(a[i])
				break;

		//cout << "10000!一共有" << maxn-i << "位" << endl;

		for(j=i;j<maxn;j++)//从非零的那一位(第i位)开始输出
			cout << a[j];
		cout << endl;
	}

	return 0;
}
上述代码,提交可以通过.


小结:

maxn (= 35660 + 1)这个数的设置会影响判题.

开始设为1000,10000,对于较小的数的阶乘可以正确计算,但是无法记录10000!的结果,所以显示PE.

后来改为20000显示PE.

再改为40000显示超时,再上网查了一下10000的阶乘有35660位.

做完题目之后,有测了一下,maxn=37000也会超时.

所以,maxn可以设置为35660 ~ 36000(区间可以更精确一点)的一个数.

posted @ 2017-03-05 20:59  tensory  阅读(495)  评论(0编辑  收藏  举报