天天看点

求一个二进制数中1的个数 popcount 算法

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

看其他博客介绍又叫

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

继续阅读