你确定会了比特位计算?

问题引入

题目:求一个数字的二进制位中有多少个1

假设我们给定一个数字为7,7的二进制为0000 0111(已省略前面的24个0)接下来我们来探究一下如何求出7的二进制当中有多少个数字1

思路一

要想求出一个数字有多少个1,我首先会想到,要是能求出这个数字的每一位数字,那么不就直接知道有多少个1了,接下来的问题就是,如何求出这个数字的每一位呢?

我们知道0 & (0/1)结果都是0,只有当1&1时结果才为1,而二进制中无非就是0和1 ,所以一个数的二进制的最后一位就可以通过&1得出来

图解

image-20220427181921847

知道了一个数二进制的最后一位之后,只每次需要将这个数进行右移1位,一共右移32次,每次&1的结果为1,计数器就+1,即可统计出这个数字一共有多少个数字1

image-20220428081541051

代码如下:

public static int Findnum(int n) {
        int count = 0;
        for (int i = 0; i < 32; i++) {
            if (((n>>i)&1)==1) {
                count++;
            }
        }
        return count;
    }
    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        int n = scanner.nextInt();
        System.out.println(Findnum(n));
    }

弊端

这样子确实可以计算出一个数字的二进制的1的个数,但是7这个数字的二进制除去后面3位是1,其余都是0,也就是说在向右移位3次之后,后面就全是0了,所以之后的29次循环就是在做无用功了,所以执行效率很低

思路二

在每进入一次循环后,就将移向右移动一位的n后的n重新赋值给n,再判断n是否为为0,要是为0,就说明此时的n的二进制全是0,就可以直接返回count,这样就可以有效的不必要的减少循环次数

public static int Findnum(int n) {
        int count = 0;
        while (n != 0) {
            if ((n& 1) == 1) {
                count++;
            }
            n = n >> 1;
        }
        return count;
    }
    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        int n = scanner.nextInt();
        System.out.println(Findnum(n));
    }

完善

但是,要是输入的数字为负数会怎么样呢?假设输入一个-1,每右移一位数字,在二进制的左端还是会补一个符号位1,所以代码就会死循环

其实只要改成无符号右移,就会在二进制的左端补0,这样就可以解决负数的问题

public static int Findnum(int n) {
        int count = 0;
        while (n != 0) {
            if ((n& 1) == 1) {
                count++;
            }
            n = n >>> 1;
        }
        return count;
    }
    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        int n = scanner.nextInt();
        System.out.println(Findnum(n));
    }

思路三

但是,还能不能再优化一下?上面的代码还是在一位一位按顺序进行移位,有没有更好的方法可以将1的个数全部快速的求出来?

其实,还是有的,要是将n每次&(n-1),那么每次就会有一个1被消去,计数器统计一下即可

image-20220428082659188

这样子每次消去一个1,效率也会增加

public static int Findnum(int n) {
        int count = 0;
        while (n != 0) {
            n = n & (n - 1);
            count++;
        }
        return count;
    }
    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        int n = scanner.nextInt();
        System.out.println(Findnum(n));
    }

拓展

给出一个数,判断它是不是2的次方

要判断是不是2的次方,就要知道2的次方的数是什么样子的

0000 0010

0000 0100

0000 1000

思路

可以看出2的次方数的二进制都只有一个1,那么就可以直接按照第一种方法在32位中统计所有的1的个数,要是个数为1,就是2的次方数,就返回1,1的个数不是1,就返回-1

 public static int multipleOfTwo(int n) {
        int count = 0;
        for (int i = 0; i < 32; i++) {
            if (((n>>i)&1)==1) {
                count++;
            }
        }
        if (count == 1) {
            return 1;
        } else {
            return -1;
        }
public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        int n = scanner.nextInt();
        System.out.println(multipleOfTwo(n));
    }

但是,还是要右移32次才能知道最终count的结果,所以效率还是不高

进一步优化

但要是借用方法三的思路,将n&(n-1),要是n是2的次方数,&(n-1)结果就一定会是0

image-20220428084924290

  public static int multipleOfTwo(int n) {
        if ((n & (n - 1)) == 0) {
            return 1;
        }
        else
            return -1;
    }
    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        int n = scanner.nextInt();
        System.out.println(multipleOfTwo(n));
    }

感谢大家的阅读,如有错误,还请大家指正。

  • 34
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 30
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 30
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值