关于二进制位运算并不是很难,总共只有五种运算:与、或、异或、左移和右移。
1:左移时,高位舍去,低位补0;
2:右移时,分有符号数和无符号数。如果是无符号数,高位直接补0;如果是有符号数,左边补符号位。
-
题目描述:
-
输入一个整数,输出该数二进制表示中1的个数。其中负数用补码表示。
-
输入:
-
输入可能包含多个测试样例。
对于每个输入文件,第一行输入一个整数T,代表测试样例的数量。对于每个测试样例输入为一个整数。
。n保证是int范围内的一个整数。
-
输出:
-
对应每个测试案例,
输出一个整数,代表输入的那个数中1的个数。
-
样例输入:
-
3 4 5 -1
-
样例输出:
-
1 2 32
题目解析:
方案一:错误的方法
既然让判断1的个数,那么就让数据右移一位,并和1相与,看是否为1来计数。代码如下:
int NumberOf1(int n)
{
int count = 0;
while(n){
if(n&1)
count++;
n = n>>1;
}
return count;
}
优点:通过右移操作符来代替除以2,可以提高程序运行效率。
缺点:如果n为负数,右移时最高位补1,会造成死循环。
方案二:逆向思维
我们可以利用与和移位操作,判断最低位的位是1还是0,来统计1的个数。但必须考虑有符号数的情况,可以利用统计移动的维数上线,比如32位。不过不同系统,int位不一定一样。这时,我们可以变通,通过让1来移位,然后相与操作来判断。
int NumOfOne1(int n)
{
int count = 0;
int flag = 1;
while(flag){
if(n & flag)
++count;
flag = flag << 1;
}
return count;
}
方案三:运用数学方法!
还有一种整数中有几个1就只循环几次的算法:
一个数不为0,那么二进制中肯定含1。当让这个数减去1时,最低位的1由1->0,更低位的0由0->1。当让n与n-1相与以后,n中最低位的1变成0。一次这样的操作,消去一个1,那么二进制中有多少个1,就循环多少次即可。
int NumOfOne(int n)
{
int count = 0;
while(n){
++count;
n = n&(n-1);
}
return count;
}
完整代码如下:
#include <stdio.h>
#include <stdlib.h>
int NumOfOne(int n);
int main(void)
{
int n,e;
while(scanf("%d",&n) == 1){
if(n<=0)
continue;
for(int i = 0;i < n;i++){
scanf("%d",&e);
printf("%d\n",NumOfOne(e));
}
}
return 0;
}
/*
int NumOfOne1(int n)
{
int count = 0;
int flag = 1;
while(flag){
if(n & flag)
++count;
flag = flag << 1;
}
return count;
}
*/
int NumOfOne(int n)
{
int count = 0;
while(n){
++count;
n = n&(n-1);
}
return count;
}