c++面试题(2)-----计算一个数字的二进制数中1的个数

  • 操作系统:ubuntu22.04
  • IDE:Visual Studio Code
  • 编程语言:C++11

题目描述:

请实现一个函数 int NumberOf1(int n),输入一个整数(可能为负数),输出该数二进制表示中 1 的个数。例如:

复制代码
把 9 表示成二进制是 1001,有 2 个 1;
把 -1 表示成二进制在补码形式下全是 1(32位系统下是 32 个 1)。

示例

bash 复制代码
输入: n = 9
输出: 2

输入: n = -1
输出: 32

解题思路

这个问题主要考察的是对 二进制和位运算的理解,以及如何处理 负数(补码表示)。

以下是几种常见解法:

方法一:逐位与操作(推荐)

cpp 复制代码
int NumberOf1(int n)
{
    int count = 0;
    while (n != 0)
    {
        ++count;
        n = n & (n - 1); // 消除最右边的一个 1
    }
    return count;
}

优点:

  • 不需要考虑正负;
  • 效率高,循环次数等于 1 的个数;
  • 利用了 n & (n - 1) 特性:每次消去一个 1。

方法二:右移判断每一位是否为 1

cpp 复制代码
int NumberOf1(int n) 
{
    int count = 0;
    unsigned int flag = 1; // 使用无符号数来避免右移时出错
    while (flag != 0)
    {
        if (n & flag)
            ++count;
        flag <<= 1;
    }
    return count;
}

说明:

  • 循环 32 次(32位整数);
  • 每次用 flag 来测试某一位是否为 1;
  • 对于负数也能正确处理(因为使用了无符号变量);

方法三:使用 C++ 内建函数(C++20 起)

如果你使用的是 C++20 或更高版本,可以使用标准库提供的函数:

cpp 复制代码
#include <bit>
int NumberOf1(int n) {
    return std::popcount(static_cast<unsigned int>(n));
}

说明:

  • std::popcount() 返回一个数中 1 的数量;
  • 需要将 int 强转为 unsigned int 来保证行为一致。

示例代码

cpp 复制代码
#include <iostream>

using namespace std;

class Solution {
public:
    int NumberOf1( int n )
    {
        int count = 0;
        while ( n != 0 )
        {
            ++count;
            n = n & ( n - 1 );  // 消除最右边的 1
        }
        return count;
    }
};

int main()
{
    Solution solution;
    cout << "9 中 1 的个数:" << solution.NumberOf1( 9 ) << endl;    // 输出 2
    cout << "-1 中 1 的个数:" << solution.NumberOf1( -1 ) << endl;  // 输出 32
    return 0;
}

运行结果

bash 复制代码
9 中 1 的个数:2
-1 中 1 的个数:32
相关推荐
重生之我是Java开发战士2 分钟前
【优选算法】前缀和:一二维前缀和,寻找数组的中心下标,除自身以外数组的乘积,和为K的子数组,和可被K整除的子数组,连续数组,矩阵区域和
线性代数·算法·矩阵
梵刹古音4 分钟前
【C语言】 循环结构
c语言·开发语言·算法
消失的旧时光-19437 分钟前
C++ 函数参数传递方式总结:什么时候用值传递、引用、const 引用?
开发语言·c++
一匹电信狗9 分钟前
【C++】CPU的局部性原理
开发语言·c++·系统架构·学习笔记·c++11·智能指针·新特性
皮皮哎哟12 分钟前
冒泡排序与数组传递全解析 一维二维指针数组及二级指针应用指南
c语言·算法·冒泡排序·二维数组·指针数组·传参·二级指针
m0_5613596713 分钟前
C++代码冗余消除
开发语言·c++·算法
会开花的二叉树18 分钟前
吃透Reactor多线程:EventLoop_Channel_ThreadPool协作原理
开发语言·c++·tcp/ip·servlet
Jm_洋洋20 分钟前
【C++进阶】虚函数、虚表与虚指针:多态底层机制剖析
java·开发语言·c++
近津薪荼24 分钟前
优选算法——滑动窗口1(单调性)
c++·学习·算法
头发还没掉光光24 分钟前
Linux 高级 IO 深度解析:从 IO 本质到 epoll全面讲解
linux·服务器·c语言·c++