求一个二进制数中1的个数

来自笔试题 注意位运算符的使用 要注意基础啊 = = 

看其他博客介绍又叫

popcount 算法

 实际应用在   汉明码 (Hamming Code) 的检验中 ,(汉明码 似曾相识啊 好像在组成原理书上有介绍 —— 讲存储器奇偶校验那节)

#include<stdio.h>
//输出二进制数中1的个数
int f1(int x)
{
	int count=0;
	for(;x;)
	{
		count+=x&1;//与运算 x & 00..01 
		x>>=1;// x/2	
	}
	return count;
}
int f2(int x)
{
	int count=0;
	for(;x;)
	{
		x&=x-1;//最右侧的 1 消除 1..110 & 1...101 =1...100
					 //同理 使用 x&(x+1) 1..100&1...101=1..100 去掉0 
		count++;
	}
	return count;
}
int f3(int x)
{
	/*
	分治思想:统计每相邻2位,4位,8位一的个数,求和
	为什么 &0x55555555(0101 0101...) 
	:abcd & (0101)=0b0d abcd>>2&(0101)=0a0c  a|b|c|d->a+b|c+d 
	*/
	//注意 必须加括号 因为单目运算符向右结合 
	//( 1&1+1&1 )==((1&1 +1)&1)
	x=x&0x55555555+((x>>1)&0x55555555);//0101
	x=x&0x33333333+((x>>2)&0x33333333);//0011
	x=x&0x0f0f0f0f+((x>>4)&0x0f0f0f0f);//00001111
	x=x&0x00ff00ff+((x>>8)&0x00ff00ff);//.....
	x=x&0x0000ffff+((x>>16)&0x0000ffff);
	return x;
}
int main()
{
	printf("%d\n%d\n%d\n",f1(192),f2(127),f3(7));
	//printf("%d",(3&1+3)&1);
	return  0;
}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值