【剑指offer】面试题10:二进制中1的个数


关于二进制位运算并不是很难,总共只有五种运算:与、或、异或、左移和右移。

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 
#include 


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;
}





你可能感兴趣的:(剑指offer)