AcWing 801. 二进制中1的个数

AcWing801. 二进制中1的个数

一、题目描述

给定一个长度为 n 的数列,请你求出数列中每个数的二进制表示中 1 的个数。

输入格式
第一行包含整数 n

第二行包含 n 个整数,表示整个数列。

输出格式
共一行,包含 n 个整数,其中的第 i 个数表示数列中的第 i 个数的二进制表示中 1 的个数。

数据范围
1n100000,0109

输入样例

5
1 2 3 4 5

输出样例:

1 1 2 1 2

二、前导知识

位运算中有两个重要性质

一、x&(x)

保留二进制下最后出现1的位置,其余位置置0

演示一下 x&(x) 的计算过程:
对于 x=10 (二进制:00001010):

  1. x 的过程:
x         = 00001010
取反(~x)  = 11110101
加1       = 11110110  (-x的最终值)

x&(-x)    = 00001010
          & 11110110
          = 00000010  (结果)

x=10 的二进制表示中,最右边的1在第2位(从右往左数,从1开始) x&(x)=2,正好是这个位置对应的值。

封装函数:

int lowbit(int x) {
    return x & (-x);
}

二、x&(x1)

消除二进制下最后出现1的位置,其余保持不变

假设 x=52,其二进制表示为:

x    = 52  = 00110100
x-1  = 51  = 00110011
x & (x-1)  = 00110000 = 48

工作原理
当我们将一个数减1时,从最右边的1开始,这一位变成0,而它右边的所有0都变成1
当我们将原数和减1后的数做与运算时,最右边的1及其右边的所有位都会变成0,而左边的位保持不变
这就是为什么这个技巧常用于:

  • 计算二进制中1的个数(每操作一次消除一个1

  • 判断一个数是否是2的幂(如果是2的幂,执行一次后就变成0

while (x) {
    x = x & (x - 1);  // 每次操作消除最右边的1
    cnt++;            // 记录操作次数,就是1的个数
}

这个方法比直接遍历32位要高效,因为它的循环次数等于二进制中1的个数,而不是固定的32次。

三、练习x&(x1)

求下面函数的返回值 (微软)

int func(x) { 
    int countx = 0; 
    while(x){ 
          countx ++; 
          x = x & (x-1); 
     } 
    return countx; 
} 

功能:将x转化为2进制,看含有的1的个数。

注: 每执行一次x=x&(x1),会将x用二进制表示时最右边的一个1变为0,因为x1将会将该位(x用二进制表示时最右边的一个1)变为0

判断一个数x是否是2n次方

#include <bits/stdc++.h>
int func(int x){
    if((x & (x-1)) == 0)
        return 1;
    else
        return 0;
}
 
int main(){
    int x = 8;
    printf("%d\n", func(x));
    return 0;
}

思路
如果一个数是2n次方,那么这个数用二进制表示时其最高位为1,其余位为0

四、利用 x&x

#include <iostream>
using namespace std;

int lowbit(int x) {
    return x & (-x);
}

int main() {
    int n;
    cin >> n;
    while (n--) {
        int x;
        cin >> x;

        int res = 0;
        while (x) x -= lowbit(x), res++;

        printf("%d ", res);
    }
    return 0;
}

五、利用 x&(x1)

#include <bits/stdc++.h>

using namespace std;

int main() {
    int n;
    cin >> n;
    while (n--) {
        int cnt = 0;
        int x;
        cin >> x;
        while (x) {
            x = x & (x - 1);
            cnt++;
        }
        printf("%d ",cnt);
    }
    return 0;
}

六、遍历每一个二进制位

#include <bits/stdc++.h>

using namespace std;

int main() {
    int n;
    cin >> n;
    while (n--) {
        int cnt = 0;
        int x;
        cin >> x;
        for (int i = 0; i < 32; i++)
            if (x >> i & 1) cnt++;
        cout << cnt << " ";
    }
    return 0;
}
posted @   糖豆爸爸  阅读(628)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 【自荐】一款简洁、开源的在线白板工具 Drawnix
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· Docker 太简单,K8s 太复杂?w7panel 让容器管理更轻松!
历史上的今天:
2017-09-08 Python3发送qq邮件,测试通过
Live2D
点击右上角即可分享
微信分享提示