码迷,mamicode.com
首页 > 其他好文 > 详细

每天一道算法题:数字二进制形式中1的个数

时间:2014-06-25 12:05:04      阅读:208      评论:0      收藏:0      [点我收藏+]

标签:style   class   blog   code   color   问题   

题目:请实现一个函数,属于一个整数,输出该数二进制表示中1的个数,例如把9表示成二进制是1001,有2位为1。因此如果输入9,该函数输出2。

  • 可能的死循环陷阱

     看完题目,相信大家很快就能想到一个解题思路:先判断整数二进制表示中最右边的一位是否为1,接着把输入的整数右移一位,此时原来处于从右边起的第二位被移动至最右边了,再判断是不是1,这样每次移动一位,直到这个整数变成0,即能够得到整数二进制表示形式中1的个数,而现在问题变为如何判断数字的最后一位为1,其实这个也很简单,只需要将数字与1做与运算,如果结果为1,则表示最后一位为1,否则为0。有了上面的思路,我们很快能写出如下的代码:

 1 int NumberOf1(int number)
 2 {
 3     int count = 0;
 4     
 5     while (0 != number)
 6     {
 7         if (0 != (number & 0x01))
 8         {
 9             ++count;
10         }
11 
12         number = number >> 1;
13     }    
14 
15     return count;
16 }

  作为一个严谨的程序猿,写完代码必须要对代码进行相应的测试,当我们输入正整数时,程序能够正确运行,但是当我们输入负整数时,程序好像不能停止了,为什么会这样?当我们再回头去看代码,并看到number = number >> 1时,才恍然大悟,因为负数带有符号位,而在进行右移操作时,由于要保持其任然为负数,会在左边的第一位补1而不是0,这样一直进行以为,最终整数变为了0XFFFFFFFF,而陷入死循环。如何避免该问题呢?

  • 常规解法

  当上面的问题出现时,我们想到,既然不能将整数向右移动,那我们为什么不将1向左边移位后再与输入整数进行与运算呢?这样的处理方式能达到同样的效果,并且左移永远都只会在右边的位置补0,这样就避免了死循环的出现,好了,有了这样的思路我们又写出下面的第二个版本代码:

 1 int NumberOf1(int number)
 2 {
 3     int count = 0;
 4     int flag = 0x01;
 5 
 6     while (0 != flag)
 7     {
 8         if (0 != (number & flag))
 9         {
10             ++count;
11         }
12 
13         flag = flag << 1;
14     }
15 
16     return count;
17 }

  当完成代码之后,进行测试都能得到正确的结果,似乎已经是完美答案了,but,我们发现不管我们输入的整数二进制形式中有几个1,我们的程序都将循环32次才能得到结果,有没有什么方法能够使我们的循环次数更少呢?比如二进制中有几个1就只循环几次的方法,答案是肯定的,请接着往下看。

  • 惊艳的解法

    再分析算法之前,我们先来分析把一个整数减1的情况,如果一个整数不等于0,那么该整数的二进制表示中至少有1位是1,先假设这个数的最后一位是1,将该整数减1之后,最后一位变为0,其他位保持不变,接下来假设最后一位不是1而是0的情况,如果该数最右边的1位于第m位,则将此数减1之后,该数的第m位由1变为0,而m位之后的所有位都由0变为1,第m位之前的位都保持不变,如果再将该整数与该整数减1之后的结果进行与运算,其效果相当于把整数最右边为1的位由1变为0。我们把上面的分析总结起来就是:把一个整数减1之后再与整数本身做与运算,会把整数二进制形式中最右边的一个1变成0,那么一个整数的二进制形式中存在多少个1,我们就可以对该整数做几次上述的操作,有了上面的思路,我们能写出新的解法:

 1 int NumberOf1(int number)
 2 {
 3     int count = 0;
 4 
 5     while (0 != number)
 6     {
 7         ++count;
 8 
 9         number = number & (number - 1);
10     }
11 
12     return count;
13 }

 

每天一道算法题:数字二进制形式中1的个数,布布扣,bubuko.com

每天一道算法题:数字二进制形式中1的个数

标签:style   class   blog   code   color   问题   

原文地址:http://www.cnblogs.com/wanxiao/p/3807147.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!