问题描述:
求二进制数中1 的个数,这个问题看似很简单,例如我们可以用不断除2(右移)来解决,这里讨论这个问题的原因在于说明其一种比较巧妙的解法(这种解法在程序员面试宝典也出现过)
代码如下:
Int Count(BYTE v)
{
Intnum = 0;
While(v)
{
V&= (v-1);
Num++;
}
Return num;
}
如果用上述代码举例说明,可以发现的确可行,可是上述代码是如何得到的?
问题分析:
试想一下,如果一个二进制数中只有1个“1”,那么这个数一定为2的幂!对于2的幂,我们已经很熟悉,2的幂减1,那么一定为全1(除最高位),例如,1000,1000-1=0111,1000&(1000-1)=0,那么对于不止一个1的二进制数a呢,a&(a-1)又为多少?假设有两个1,例如1010。
套用1个1的性质,1010=1000+10!那么可以发现,a&(a-1)=(1000+10)&(1000+10-1),关于&运算,没有结合律之说,但是对于(1000+10)&(1000+10-1),可以“感觉”,最后的答案为1000,因为1000只会影响10与10-1的高位,因此,最后的计算只计算10&(10-1)=0,加上高位,最后的结果为1000,在对1000做1000&(1000-1)为0,对于两个1,要两次v&(v-1),就可以时v归0,同理n个1,则n次v&(v-1),因此得到上面的算法。