剑指offer-11、⼆进制中1的个数

题⽬描述

输⼊⼀个整数,输出该数 32 位⼆进制表示中 1 的个数。其中负数⽤补码表示。

示例1 输⼊:10 返回值:2 说明:⼗进制中10的32位⼆进制表示为0000 0000 0000 0000 0000 0000 0000 1010,其中有两个1。

示例2 输⼊:-1 返回值:32 说明:负数使⽤补码表示 ,-1的32位⼆进制表示为1111 1111 1111 1111 1111 1111 1111 1111,其中32个1

思路及解答

错误解答(不考虑负数)

⾸先说⼀个错误的解法,很多⼈可能会想到,那就是不断对 2 取余数。但是这种做法有个致命的缺陷,那就是忽略了负数,负数使⽤补码表示的时候,是取反之后加⼀

java 复制代码
public class Solution {
	public int NumberOf1(int n) {
		int num = 0;
		while (n != 0) {
			int tmp = n % 2;
			if (tmp == 1||tmp==-1) ++num;
			n /= 2;
		}
		return num;
	}
}

字符串遍历

将整数转换为二进制字符串,然后遍历字符串统计'1'的个数。

java 复制代码
public int NumberOf1(int n) {
    String binaryStr = Integer.toBinaryString(n);
    int count = 0;
    for (int i = 0; i < binaryStr.length(); i++) {
        if (binaryStr.charAt(i) == '1') {
            count++;
        }
    }
    return count;
}
  • 时间复杂度:O(n),n为二进制位数(固定32位)
  • 空间复杂度:O(n),需要存储二进制字符串

API调⽤(不推荐)

Java标准库提供了统计二进制1个数的方法。

java 复制代码
public int hammingWeight(int n) {
    return Integer.bitCount(n);
}

位掩码与移位(逐位检查)

使用位掩码和移位操作逐位检查是否为1。

移位算法,把整数当成⼆进制,不断向右移位和1 进⾏与计算。利⽤了所有数和1 进⾏与计算,结果为1则证明最后⼀位是1 。

java 复制代码
public int NumberOf1(int n) {
    int count = 0;
    int mask = 1;
    for (int i = 0; i < 32; i++) {
        if ((n & mask) != 0) {
            count++;
        }
        mask <<= 1;
    }
    return count;
}

无符号右移:

java 复制代码
public int NumberOf1(int n) {
    int count = 0;
    while (n != 0) {
        count += (n & 1);
        n >>>= 1; // 无符号右移
    }
    return count;
}

位运算

利用n & (n - 1)可以消除最右边的1的特性,直到n变为0

⽐如7的⼆进制是111 ,那么7&6=111&110=110=6 ,就完美把最后⼀位1 变成0 了, 6 的⼆进制是110 , 6&5=110&101=100=4 ,也把最后⼀位1 变成了0 。

负数呢?⽐如: 32位-7 = 11111111111111111111111111111001 , 32位-8 = 11111111111111111111111111111000, -7&-8 = 11111111111111111111111111111000

java 复制代码
public class Solution {
    public int NumberOf1(int n) {
        int num = 0;
        while (n != 0) {
            num++;
            n &= (n - 1);
        }
        return num;
    }
}
相关推荐
万岳科技系统开发3 分钟前
食堂采购系统源码库存扣减算法与并发控制实现详解
java·前端·数据库·算法
独断万古他化6 分钟前
【Spring 原理】Bean 的作用域与生命周期
java·后端·spring
*小海豚*12 分钟前
在linux服务器上DNS正常,但是java应用调用第三方解析域名报错
java·linux·服务器
撩得Android一次心动27 分钟前
Android LiveData 全面解析:使用Java构建响应式UI【源码篇】
android·java·android jetpack·livedata
组合缺一31 分钟前
Solon AI (Java) v3.9 正式发布:全能 Skill 爆发,Agent 协作更专业!仍然支持 java8!
java·人工智能·ai·llm·agent·solon·mcp
MSTcheng.36 分钟前
【C++】C++11新特性(二)
java·开发语言·c++·c++11
一 乐39 分钟前
校园二手交易|基于springboot + vue校园二手交易系统(源码+数据库+文档)
java·数据库·vue.js·spring boot·后端
KIKIiiiiiiii40 分钟前
微信个人号API二次开发中的解决经验
java·人工智能·python·微信
80530单词突击赢41 分钟前
SpringBoot整合SpringMVC全解析
java·spring boot·后端
vx1_Biye_Design1 小时前
基于Spring Boot+Vue的学生管理系统设计与实现-计算机毕业设计源码46223
java·vue.js·spring boot·spring·eclipse·tomcat·maven