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

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

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值