位运算--C语言版

简介: 位运算--C语言版

问题引入
题目:求一个数字的二进制位中有多少个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));
}
目录
相关文章
|
7月前
|
算法 程序员 C语言
C语言位运算
C语言位运算
57 0
|
7月前
|
算法 测试技术 C语言
【C语言】异或(^)操作符
【C语言】异或(^)操作符
110 0
|
C语言 C++
C语言:算数转换
已知:char 和 short 这两种字节长度小于 int 类型参与算术运算时,会进行整型提升。 而当字节长度大于 int 类型的数据参与运算时,如果某个操作符的两个操作数是不同类型,其中一个操作数需要按级别(级别低的数据转换为级别高的数据)转换为另一个操作数的类型,这样的转换即为算数转换
80 1
|
6月前
|
C语言
C语言中的位运算详解
C语言中的位运算详解
140 0
|
7月前
|
程序员 编译器 C语言
C语言中的位运算技术详解
C语言中的位运算技术详解
114 1
|
存储 机器学习/深度学习 传感器
|
存储 编译器 C语言
【C语言】数据以及位运算
【C语言】数据以及位运算
89 0
【C语言】带你回顾C语言全部运算符(详解)
前言 ​ 🌞🌞想要学好C语言,操作符是非常重要且必不可少的一部分内容,操作符分为算数运算符、移位操作符、位操作符、赋值操作符、单目操作符、关系操作符、逻辑操作符、条件操作符、逗号表达式、下标引用,函数调用和结构成员。 操作符的基本特性 每个操作符都有自己的语义 每个操作符都有对应的优先级,如果两个操作符挨在一块,先执行优先级高的操作符,如果优先级相同,取决于他们的结合性。 每个操作符都有结合性。 操作符的结合性定义了操作符对操作数执行操作的顺序,例如:右结合性表示该操作符对其操作数从右向左执行操作。
|
存储 算法 编译器
【符号】 — C语言版
【符号】 — C语言版
114 0
【符号】 — C语言版
|
存储 C语言
C语言入门系列之12.位运算(下)
位运算是指按二进制位进行的运算,这是因为在系统软件中,常要处理二进制位的问题。
C语言入门系列之12.位运算(下)