重新开始战斗08-编程之美-求二进制数中1的个数

问题描述:

求二进制数中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),因此得到上面的算法。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值