剑指 Offer(第2版)面试题 15:二进制中1的个数

剑指 Offer(第2版)面试题 15:二进制中1的个数

  • [剑指 Offer(第2版)面试题 15:二进制中1的个数](#剑指 Offer(第2版)面试题 15:二进制中1的个数)

剑指 Offer(第2版)面试题 15:二进制中1的个数

题目来源:26. 二进制中1的个数

此题与 Leetcode191. 位1的个数 一致。

解法1:位运算

循环检查给定整数 n 的二进制位的每一位是否为 1。

当检查第 i 位时,我们可以让 n 与 2i 进行与(&)运算,当且仅当 n 的第 i 位为 1 时,运算结果不为 0。

设置一个计数器,遍历一次即可得到位 1 的个数。

注意:1 << i 必须是 uint32_t 的。

代码:

c 复制代码
class Solution
{
public:
	int NumberOf1(int n)
	{
		int count = 0;
		for (int i = 0; i < 32; i++)
			if (n & (uint32_t)1 << i)
				count++;
		return count;
	}
};

复杂度分析:

时间复杂度:O(k),其中 k = 32,是 int 的位数。

空间复杂度:O(1)。

解法2:n & (n - 1)

随便一个 n:

那么 n - 1 会发生什么:对于上面随便给的一个 n,减去 1,但是低 2 位都是 0,减 1 必然需要借位,那何时结束呢?答案就是遇到 1 的时候。如果第一步骤中给的 n 末尾为 1,那么就直接在低位就结束了,也符合遇到 1 就停止的规则。

总结而言,n - 1 操作:

  1. 无论是借位,还是减 1,遇到 1 就停止;

  2. 从低位到高位,一直到 1,每一位都发生了反转。

n & (n - 1):将 n 最低位的一个 1 变成 0。

于是,要求一个数 n 中二进制 1 的个数,我们可以不断地对 n 进行 n = n & (n - 1) 操作,每进行一次,n 中最低位的 1 就会变成 0,直到 n 变成 0 为止,操作次数就是 n 中二进制 1 的个数。

代码:

c 复制代码
class Solution
{
public:
	int NumberOf1(int n)
	{
		int count = 0;
		while (n)
		{
			n = n & (n - 1);
			count++;
		}
		return count;
	}
};

复杂度分析:

时间复杂度:O(log n)。

空间复杂度:O(1)。

相关题目

LeetCode 231. 2 的幂

相关推荐
uhakadotcom19 小时前
Next.js 从入门到精通(1):项目架构与 App Router—— 文件系统路由与目录结构全解析
前端·面试·github
chbmvdd19 小时前
week5题解
数据结构·c++·算法
用户120391129472619 小时前
面试官最爱问的字符串反转:7种JavaScript实现方法详解
算法·面试
vir0220 小时前
小齐的技能团队(dp)
数据结构·c++·算法·图论
月夜的风吹雨20 小时前
【C++红黑树】:自平衡二叉搜索树的精妙实现
开发语言·c++·红黑树
讨厌下雨的天空20 小时前
Linux信号
linux·运维·c++
赖small强20 小时前
【Linux C/C++开发】第26章:系统级综合项目理论
linux·c语言·c++
南山安20 小时前
从反转字符串看透面试官的“内心戏”:你的算法思维到底怎么样?
javascript·算法·面试
fpcc21 小时前
跟我学C++中级篇——重载问题分析之函数模板重载的问题
c++
仟濹21 小时前
【C/C++】经典高精度算法 5道题 加减乘除「复习」
c语言·c++·算法