P2529 [SHOI2001]击鼓传花
题意:求出(n!)末尾最后一位非0数字
数据范围:(n<=10^{100})
我们从简单的开始考虑
1.显然,(n!)可以被这么表示
(n!=c imes 2^a imes 5^b)
显然有(a>b)
2.末尾的元素即为(c \%10 imes 2^{a-b} \%10)
显然这个复杂度是我们所不能接受的
我们发现(5)很特殊
我们把所有(5)的倍数都取出来(注意取出的是(5)的倍数而不是因数(5)),给每个(5)配一个(2)
相当于把(5*1,5*2,5*3,...,5*n)中的(5)约去,发现剩下的是一个规模1/5的相同子问题
令(fac(i))表示(i!)的末尾非0数字,则有(fac(i)=fac(lfloor i/5 floor) imes ? \% 10)
我们想办法求出(?)的贡献
因为因子(2)消不完,所以末尾必须是偶数
发现剩下的数可以直接(mod) (10)了
把每(20)位分成一块,对末位的贡献为(6)
因为$6 imes $任意偶数,末尾不变
所以答案只需要把20位以内的额外贡献找到就可以
我们考虑直接把这20位打表,然后递归处理子问题
复杂度可以接受(高精度我不会算)
Code:
#include <cstdio>
#include <cstring>
const int N=102;
struct l_num
{
int num[N];
l_num()
{
memset(num,0,sizeof(num));
}
l_num(char c[])
{
memset(num,0,sizeof(num));
num[0]=strlen(c);
for(int i=1;i<=num[0];i++)
num[i]=c[num[0]-i]-'0';
}
l_num friend operator /(l_num n1,int n2)
{
for(int i=n1.num[0];i>1;i--)
{
n1.num[i-1]+=n1.num[i]%n2*10;
n1.num[i]/=n2;
}
n1.num[1]/=n2;
if(!n1.num[n1.num[0]]) --n1.num[0];
return n1;
}
int friend operator %(l_num n1,int n2)
{
return n1.num[0]==1?n1.num[1]:(n1.num[2]&1)*10+n1.num[1];
}
};
int init[21]={1,1,2,6,4,2,2,4,2,8,4,4,8,4,6,8,8,6,8,2,6};
char c[103];
int cal(l_num fac)
{
return fac.num[0]?init[fac%20]*cal(fac/5)%10:1;
}
int main()
{
int t=5;
while(t--)
{
scanf("%s",c);
l_num fac(c);
printf("%d
",cal(fac));
}
return 0;
}
2018.8.9