【剑指offer】二进制中1的个数&&2的幂

简介: 【剑指offer】二进制中1的个数&&2的幂

👉位运算👈


位运算是把数字将二进制表示之后,对每一位上0或者1的运算。二进制及其位运算是现代计算机学科的基石,很多底层技术都离不开位运算,因此与位运算相关的题目也经常出现在面试中。因为在我们的日常生活中,用得最多的进制就是十进制,所以很多人面对二进制及位运算的题目时多少会有点不适应。但是无需害怕,如果对位运算还不太熟悉的小伙伴,可以先看一下这篇文章:👉操作符详解👈。


👉二进制中1的个数👈


题目:

请实现一个函数,输入一个整数,输出该数二进制表示中1的个数。例如:把9表示成二进制是1001,有2位是1,。因此如果输入9,该函数的返回值为2。


可能引起死循环的解法


这是一道很基本的考查二进制和位运算的面试题,题目不是很难。基本思路:先判断整数二进制表示中最右边一位是不是1,借助把输入的整数右移一位,此时原来处于从右边数起的第二位被移动到最右边了,在判断是不是1.这样每次移动一位,直到整个整数编程0为止。


现在的问题变成了判断一个整数的最右边是不是1。这很简单,只要把整数和1做位与运算看结果是不是1就知道了。1除了最右边的一位之外所有位都是0.如果一个整数和1做位与运算的结果是1,表示该整数最右边一位是1,否则就是0。


int hammingWeight(int n) 
{
    int count = 0;
    while (n)
    {
        if (n & 1)
        {
            count++;
        }
        n >>= 1;
    }
    return count;
}

19661c25e70c4547aacccf835f1bd0c0.png


这个算法好像没有什么问题啊,那么为什么通不过测试呢?原因就是,如果给上面的函数输入一个负数就会陷入死循环了。比如:0x80000000,把负数0x80000000右移一位的时候,并不是简单地把最高位的1移到后一位变成0x40000000,而是0xC0000000.这是以内移位前是个负数,仍然要保证移位后是个负数,因此移位后高位补1。如果一直做右移运算,最终这个数字就会变成0xFFFFFFFF而陷入死循环。


不能我们稍作修改,就能够通过测试了。就是把 hammingWeight 函数的参数类型改成 unsigned int。尽管你把负数传过来,hammingWeight 函数也会把这个数当做无符号整数(正数)。

7f2b550327fa475db145452a7a53d19d.png

还有一点需要注意的是:把一个整数右移一位和把一个整数除以2在数学上是等价的,但是除法的效率比移位运算要低很多,在实际编程中华应尽可能地用移位运算符来代替乘除法。


除了上面的解法,还有没有其他解法呢?其实还有,接下来为大家介绍另一种解法。


常规解法


既然,移动输入的数字n有可能造成死循环。那我们可不可以换个方向,换成移动1呢?其实也是可以的。首先把n和1做位与运算,判断n的最低位是不是为1。接着把1左移一位得到2,再和n做位与运算,就能判断n的次低位是不是1。如此反复左移,每次都能判断你的其中一位是不是1,直至1左移变成0。


int hammingWeight(int n)
{
    int count = 0;
    unsigned int flag = 1;
    while (flag)
    {
        if (n & flag )
        {
            count++;
        }
        flag <<= 1;
    }
    return count;
}

0f44f2de43e3477689a0fce250937be0.png

奇妙的解法


在分析这种算法之前,我们先来分析一个数减去1的情况。如果一个整数不等于0,那么该整数的二进制表示至少有一位是1。先假设这个数的最右边一位是1,那么减去1时,最后一位变成0而其它所有位都保持不变。也就是相当于最后一位做了取反操作,由1变成了0。


接下来假设最后一位不是1而是0的情况。如果该整数的二进制表示中最右边的1在第m位,那么减去1时,第m位由1变成0,而第m位之后的所有0都变成1,整数中第m位之前的所有位都保持不变。举个例子:一个二进制数1100,它的第三位是从最右边数起的一个1。减去1后,第三位变成0,它的后面的两位0变成1,而前面的1保持不变,因此得到的结果是1011。


在前面两种情况中,我们可以发现一个整数减去1,都是把最右边的1变成0.如果它的右边还有0的话,所以的0都变成1,而他左边所有位都保持不变。接下来我们把一个整数和它减去1的结果做位与运算,这个操作相当于把它最右边的1变成0.以前面的二进制数1100为例,它减去1的结果是1011.我们再把1100和1011做位与运算,得到的结果是1000。我们把1100最右边的1变成0,结果刚好就是1000。


总结:


把一个整数减去1,再和原整数做位与运算,会把该整数最右边的一个1变成0。那么一个整数的二进制表示中有多少个1,就可以进行多少次这样的操作。


int hammingWeight(unsigned int n)
{
    int count = 0;
    while (n)
    {
        count++;
        n = n & (n - 1);
    }
    return count;
}

ea4bbfec16b7450eba09e54062e92f25.png


👉2的幂👈


给你一个整数 n,请你判断该整数是否是 2 的幂次方。如果是,返回 true ;否则,返回 false 。

如果存在一个整数 x 使得 n == 2x ,则认为 n 是 2 的幂次方。

示例 1:

输入:n = 1

输出:true

解释:2^0 = 1


示例 2:

输入:n = 3

输出:false


提示:

  • -2^31 <= n <= 2^31 - 1


思路:一个整数如果是2的整数次方,那么它的二进制表示中有且只有一位是1,而其它所有位都是0。根据前面的分析,把这个整数减去1之后再和它自己做位与运算,这个整数最右边的1就会变成0。我们只需判断该整数的二进制表示中是不是只有一个1就行了。


int count_bit(unsigned long n)
{
    int count = 0;
    while (n)
    {
        count++;
        n = n & (n - 1);
    }
    return count;
}
bool isPowerOfTwo(int n)
{
    if (count_bit(n) == 1)
        return true;
    else
        return false;
}

52bb4ddd9823444facec60e91fb5ccf0.png


注意:之所以count_bit函数的是参数是 unsigned long,是因为不将参数设置成 unsigned long不能通过测试。


👉总结👈


本文主要讲解了如何计算二进制中1的个数,然后引出了一种巧妙的方法 n & (n-1) 来计算二进制中1的个数,并借助这种方法来判断一个数是不是2的幂。如果大家觉得有收获的话,可以点个三连支持一下!谢谢大家啦!!!💖💝❣️

相关文章
overflow滚动条样式修改
overflow滚动条样式修改
91 1
|
6月前
|
算法 机器人 数据安全/隐私保护
四自由度SCARA机器人的运动学和动力学matlab建模与仿真
本课题深入研究SCARA机器人系统,提出其动力学与运动学模型,并基于MATLAB Robotics Toolbox建立四自由度SCARA机器人仿真对象。通过理论结合仿真实验,实现了运动学正解、逆解及轨迹规划等功能,完成系统实验和算法验证。SCARA机器人以其平面关节结构实现快速定位与装配,在自动生产线中广泛应用,尤其在电子和汽车行业表现优异。使用D-H参数法进行结构建模,推导末端执行器的位姿,建立了机器人的运动学方程。
|
计算机视觉 异构计算
【YOLOv8改进-SPPF】 AIFI : 基于注意力的尺度内特征交互,保持高准确度的同时减少计算成本
YOLOv8专栏介绍了该系列目标检测框架的最新改进与实战应用。文章提出RT-DETR,首个实时端到端检测器,解决了速度与精度问题。通过高效混合编码器和不确定性最小化查询选择,RT-DETR在COCO数据集上实现高AP并保持高帧率,优于其他YOLO版本。论文和代码已开源。核心代码展示了AIFI Transformer层,用于位置嵌入。更多详情见[YOLOv8专栏](https://blog.csdn.net/shangyanaf/category_12303415.html)。
|
机器学习/深度学习 Java 开发工具
【能力展现】魔改ZXING源码实现商业级DM码检测能力
【能力展现】魔改ZXING源码实现商业级DM码检测能力
535 1
|
10月前
|
网络协议
深入解析:TCP四次挥手断开连接的全过程及必要性
在网络通信中,TCP(传输控制协议)以其可靠性和顺序保证而闻名。然而,TCP连接的建立和终止同样重要,它们确保了网络资源的有效管理和数据传输的完整性。本文将详细描述TCP连接的四次挥手过程,并探讨为何需要四次挥手来正确终止一个TCP连接。
319 2
|
10月前
|
存储 文件存储 数据库
【QT项目】QT项目综合练习之简易计数器(QT6+文件存储)
【QT项目】QT项目综合练习之简易计数器(QT6+文件存储)
183 2
|
11月前
什么是复数
【10月更文挑战第12天】什么是复数
1582 1
|
Serverless 应用服务中间件 网络安全
函数计算操作报错合集之如何处理报错 "Function instance health check failed on port 7860 in 120 seconds."
在使用函数计算服务(如阿里云函数计算)时,用户可能会遇到多种错误场景。以下是一些常见的操作报错及其可能的原因和解决方法,包括但不限于:1. 函数部署失败、2. 函数执行超时、3. 资源不足错误、4. 权限与访问错误、5. 依赖问题、6. 网络配置错误、7. 触发器配置错误、8. 日志与监控问题。
184 1

热门文章

最新文章