题目:请实现一个函数,输入一个整数。输出该数二进制表示中1的个数。

比如把9表示成二进制是1001,有2位是1。因此假设输入9。该函数输出2.

1、可能引起死循环的解法

这是一道非常主要的考察二进制和位运算的面试题。

题目不是非常难。面试官提出问题之后,我们非常快形成一个主要的思路:先推断证书二进制表示中最右边一位是不是1.接着把输入的证书右移一位,此时原来处于从右边树起的第二位被移到最后一位,再推断是不是1.这样没移动一位,知道整个整数变成0为止。如今的问题变成怎么推断一个整数的最右边是不是1了。

这非常easy,仅仅要把整数和1做位与运算看结果是不是0就知道了。1除了最右边的一位之外全部的位都是0.基于这个思路,我们非常快写出这种代码:

int numberOf1(int n)

{

int count = 0;

while(n!=0){

if(n & 1)

count++;

n = n>>1;

}

return count;

}

面试官看了 代码后可能会问:把证书右移一位和把整数除以2在数学上是等价的,那上面的代码中能够把右移换成除以2吗?答案是否定的。由于除法的效率比移位运算要低非常多,在实际编程中应尽可能地用移位运算取代乘除法。

面试官会问第二个问题就是:上面的函数假设输入一个负数,比方0x80000000,执行的时候会发生什么情况呢?把负数0x80000000右移一位的时候,并非简单地把最高位的1移到第二位变成0x40000000,而是0xC0000000.这是由于移位前是个负数,仍然保证移位是个负数,因此移位后的最高位会设为1.假设一直做右移位运算,终于这个数字会编程0xFFFFFFFF而陷入死循环。

2、常规解法:

为了避免死循环,我们能够不右移输入的数字n.首先把n和1做与运算,推断n的最低位是不是1.接着把1左移一位得到2,再和n做与运算,就能推断n的次低位是不是1。。。

这样重复左移,每次都能推断n的当中一位是不是1.基于这种思路,我们能够写出这种代码:

int number1(int n){

int count = 0;

int flag= 1;

while(flag ! =0){

if(n& flag)

count++;

flag =flag <<1;

}

return count;

}

这个解法中循环的次数等于二进制中的位数,32位的整数须要循环32次,以下我们再介绍一个算法。整数中有几个1就仅仅循环几次。

3、能给面试官带来惊喜的算法。

我们的分析就是:把一个整数减去1。再和原整数做与运算,会把该整数最右边的一个1变成0.那么一个整数的二进制表示中有多少个1,就能够进行多少次运算。基于这种思路。我们能够写出这种代码:

/**
*题目:实现一个函数,输入一个整数,输出该数二进制表示中的1的个数。 *比如把9改成二进制是1001,有2位是1.因此假设输入9。该函数输出2.
*/
package swordForOffer; /**
* @author JInShuangQi
*
* 2015年7月30日
*/
public class E10NumberOf1InBinary {
public int numberOf1(int num) {
int count = 0;
while (num != 0) {
count++;
num = num & (num - 1); }
return count;
} public static void main(String[] args) {
E10NumberOf1InBinary test = new E10NumberOf1InBinary();
System.out.println(test.numberOf1(9));
}
}

剑指Offer面试题10(Java版):二进制中的1的个数的更多相关文章

  1. 剑指offer面试题14(Java版):调整数组顺序使奇数位于偶数的前面

    题目:输入一个整数数组.实现一个函数来调整该数组中数字的顺序.使得全部奇数位于数组的前半部分.全部偶数位于数组的后半部分. 1.基本实现: 假设不考虑时间复杂度,最简单的思路应该是从头扫描这个数组,每 ...

  2. 【剑指Offer面试题】 九度OJ1371:最小的K个数

    题目链接地址: http://ac.jobdu.com/problem.php?pid=1371 题目1371:最小的K个数 时间限制:1 秒内存限制:32 兆特殊判题:否提交:5938解决:1265 ...

  3. 剑指offer面试题3 二维数组中的查找(c)

    剑指offer面试题三:

  4. 剑指offer 面试题43. 1~n整数中1出现的次数

    leetcode上也见过一样的题,当时不会做 看了一下解法是纯数学解法就没看,结果剑指offer上也出现了这道题,那还是认真看下吧 对于数字abcde,如果第一位是1,比如12345,即计算f(123 ...

  5. 剑指offer面试题3二维数组中的查找

    题目: 在一个二维数组中,每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序.请完成一个函数,输入这样的一个二维数组和一个整数,判断数组中是否含有该整数. 需要与面试官确认的是,这 ...

  6. 剑指offer——面试题10:斐波那契数列

    个人答案: #include"iostream" #include"stdio.h" #include"string.h" using na ...

  7. 剑指offer面试题3 二维数组中的查找 (java)

    注:java主要可以利用字符串的length方法求出长度解决这个问题带来方便 public class FindNum { public static void main(String[] args) ...

  8. 剑指Offer第36题—Java版

    本题使用归并排序的思想,结合归并排序,写出的算法解. //数组中的逆序对 public static int InversePairs(int[] array){ if(array==null||ar ...

  9. 剑指offer 面试题10.2:青蛙变态跳台阶

    题目描述 一只青蛙一次可以跳上1级台阶,也可以跳上2级--它也可以跳上n级.求该青蛙跳上一个n级的台阶总共有多少种跳法. 编程思想 因为n级台阶,第一步有n种跳法:跳1级.跳2级.到跳n级跳1级,剩下 ...

随机推荐

  1. 【hdu 3478】Catch

    [Link]:http://acm.hdu.edu.cn/showproblem.php?pid=3478 [Description] 一个人从起点s出发,假设他在时间t在节点x; 则在时间t+1,他 ...

  2. [Vue + TS] Create Type-Safe Vue Directives in TypeScript

    Directives allow us to apply DOM manipulations as side effects. We’ll show you how you can create yo ...

  3. ubuntu-工作环境配置(van)

    我们平时用到的工具主要vim,retag_app,提交代码,以及一些常用命令,他们主要对应一下几个文件 1.vim ->.vimrc 2.代码提交->gitconfig 3.常用命令-&g ...

  4. 洛谷P1876 开灯

    题目背景 该题的题目是不是感到很眼熟呢? 事实上,如果你懂的方法,该题的代码简直不能再短. 但是如果你不懂得呢?那...(自己去想) 题目描述 首先所有的灯都是关的(注意是关!),编号为1的人走过来, ...

  5. input表单验证(全面)

    1.英文字母 1 <script type="text/javascript"> 2 //验证只能是字母 3 function checkZm(zm){ 4 var z ...

  6. 9 abstract 和 Virtual 之间的差别

    (1) abstract方法没有详细的实现.同一时候必须被覆写 (2) 虚(Virtual)方法能够没有详细的实现,也不一定必须覆写(虚方法定义时,能够没有详细的实现代码,可是必须创建方法体:即必须有 ...

  7. 管理aix的密码策略

    aix 中 /etc/security/user 存放用户的概要 常用参数参数如下 1.account_locked      defines whether the account is locke ...

  8. 去哪网实习总结:用到的easyui组件总结(JavaWeb)

    本来是以做数据挖掘的目的进去哪网的,结构却成了系统开发... 只是还是比較认真的做了三个月,老师非常认同我的工作态度和成果.. . 实习立即就要结束了,总结一下几点之前没有注意过的变成习惯和问题,分享 ...

  9. 关于idea新建子目录时往父目录名字后叠加而不是树形结构的解决方法(转)

    我们在IDEA中创建子目录时,子目录总是在父目录后面叠加而不是树形,如下 我们可以打开项目窗口的右上角的设置标志, 将红圈选项的√先去掉,创建好子目录后再将它选中就可以

  10. POJ3171 Cleaning Shifts DP,区间覆盖最值

    题目大意.N个区间覆盖[T1,T2]及相应的代价S,求从区间M到E的所有覆盖的最小代价是多少. (1 <= N <= 10,000).(0 <= M <= E <= 86 ...