问题1描述:
给定整数N,N求N!末尾有几个0。例如N=10,N!=3628800,N!末尾有两个0
分析:
一、将N!分解质因数,N!=(2^x)*(3^y)*(5^z)……,由于10=2*5,所以问题也就是求M=min(x,z),容易看出x大于等于z,所以也就是求z
#include <stdio.h>
int main(){
int N=10;
int ret=0;
for(int i=1;i<=N;i++){
int j=i;
while(j%5==0){
ret++;
j/=5;
}
}
printf("%d\n",ret);
return 0;
}
二、利用公式z=[N/5]+[N/5^2]+[N/5^3]+……
#include <stdio.h>
int main(){
int N=10;
int ret=0;
while(N){
ret+=N/5;
N/=5;
}
printf("%d\n",ret);
return 0;
}
问题2描述:
求N!的二进制表示中最低位1的位置
分析:
问题2与问题1的区别只是进制上的不同,对于问题1,答案其实就是N!含有多少个10,那么对于问题2答案就是N!含有多少个2了。
一、利用公式:N/2+N/4+N/16+……
#include <stdio.h>
int main(){
int N=10;
int ret=0;
while(N){
N>>=1;
ret+=N;
}
printf("%d\n",ret);
return 0;
}
二、对N/2+N/4+N/16+……发现,假设N=11011,
即:1101+110+11+1=(1000+100+1)+(100+10)+(10+1)+1
=(1000+100+10+1)+(100+10+1)+1=1111+111+1
=(10000-1)+(1000-1)+(10-1)+(1-1)=11011-(N二进制表示中1的个数)
#include <stdio.h>
int main(){
int N=10;
int ret=0,t=N;
while(t){
t&=(t-1);
ret++;
}
printf("%d\n",N-ret);
return 0;
}