编程之美:不要被阶乘吓倒

问题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;  
}  

    原文作者:_王川
    原文地址: https://blog.csdn.net/starcuan/article/details/21248131
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞