来自笔试题 注意位运算符的使用 要注意基础啊 = =
看其他博客介绍又叫
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;
}