AcWing 801. 二进制中1的个数
. 二进制中的个数
一、题目描述
给定一个长度为 的数列,请你求出数列中每个数的二进制表示中 的个数。
输入格式
第一行包含整数 。
第二行包含 个整数,表示整个数列。
输出格式
共一行,包含 个整数,其中的第 个数表示数列中的第 个数的二进制表示中 的个数。
数据范围
输入样例:
5
1 2 3 4 5
输出样例:
1 1 2 1 2
二、前导知识
位运算中有两个重要性质:
一、
保留二进制下最后出现的的位置,其余位置置
演示一下 的计算过程:
对于 (二进制:):
- 求 的过程:
x = 00001010
取反(~x) = 11110101
加1 = 11110110 (-x的最终值)
x&(-x) = 00001010
& 11110110
= 00000010 (结果)
的二进制表示中,最右边的在第位(从右往左数,从开始) ,正好是这个位置对应的值。
封装函数:
int lowbit(int x) {
return x & (-x);
}
二、
消除二进制下最后出现的位置,其余保持不变
假设 ,其二进制表示为:
x = 52 = 00110100
x-1 = 51 = 00110011
x & (x-1) = 00110000 = 48
工作原理:
当我们将一个数减时,从最右边的开始,这一位变成0,而它右边的所有都变成
当我们将原数和减后的数做与运算时,最右边的及其右边的所有位都会变成,而左边的位保持不变
这就是为什么这个技巧常用于:
-
计算二进制中的个数(每操作一次消除一个)
-
判断一个数是否是的幂(如果是的幂,执行一次后就变成)
while (x) {
x = x & (x - 1); // 每次操作消除最右边的1
cnt++; // 记录操作次数,就是1的个数
}
这个方法比直接遍历位要高效,因为它的循环次数等于二进制中的个数,而不是固定的次。
三、练习
求下面函数的返回值 (微软)
int func(x) {
int countx = 0;
while(x){
countx ++;
x = x & (x-1);
}
return countx;
}
功能:将转化为进制,看含有的的个数。
注: 每执行一次,会将用二进制表示时最右边的一个变为,因为将会将该位(用二进制表示时最右边的一个)变为。
判断一个数是否是的次方
#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;
}
思路
如果一个数是的次方,那么这个数用二进制表示时其最高位为,其余位为。
四、利用
#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;
}
五、利用
#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;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 【自荐】一款简洁、开源的在线白板工具 Drawnix
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· Docker 太简单,K8s 太复杂?w7panel 让容器管理更轻松!
2017-09-08 Python3发送qq邮件,测试通过