关于位运算的一道试题校招笔试题

之前参加某公司的笔试有一道题至今还记得,一直在寻找最优解法,今天又看到了一个更好的方法。

题目具体的不记得了,大概就是输入一个char类型的数,然后统计二进制表示中1的个数。

我当时的解法很二逼,从n = 0~7,算2的n次幂,然后拿输入的数与2的n次幂做“与”运算,结果不等于0就计数+1.很明显,这么写是不会拿到offer的。

后来我想了想,问题出在求2的n次幂上,于是就定义了一个新的变量n = 1;拿输入的数与n做“与”运算,然后n<<1,这个写在一个循环里,如下:

int NumOf1(char c)
{
    int count = 0;
    uint32_t flag = 1;
    while (flag) {
        if (c & flag)
            count++;
        flag = flag << 1;
    }
    
    return count;
}

后来想到了用C++的标准库来实现(PS:我忘记了当时是不是要求用C)

把char放到bitset里

int NumOf1(char c)
{
    bitset<8> cbet(c);
    int count = (int)cbet.count();
    return count;
}

今天看《剑指offer》又看到了新的解法,它给出了3种

第一种跟我上面讲到的移位解法差不多,不过不是将新定义的变量flag移位,而是对原输入的char移位。实现如下:

int NumOf1(char c)
{
    int count = 0;
    while (c) {
        if (c & 1)
            count++;
        c = c >> 1;
    }
    
    return count;
}

看上去似乎没有什么问题,实际上 考虑一下,当输入一个负数的时候是会引起死循环的,因为负数移位之后前面空缺的位置补1,不仅结果不正确,而且c不会为假,一直循环下去。

(PS:算数右移:有符号数 左边补符号位,即正的补0,负的补1;无符号数 左边补0;

             逻辑右移:左边总是补0

             左移操作:右端补0)


书上给出的第二种解法其实跟我上面的移位的解法是一样的。同样的实现就不再说了。

第三种解法是“能给面试官带来惊喜的解法”,先看代码:

int NumOf1(char c)
{
    int count = 0;
    while (c) {
        count++;
        c = (c - 1) & c;
    }
    
    return count;
}

这个方法乍一看不太好理解,下面就来具体分析一下:

一个数减去1,然后和它自身做“与”运算 会把从右边起第一个不是1的位置0,并且其它位保持不变。一个数减去1后,从右边起第一个是1的位就变成了0,它左边的保持不变,右边的0都变成了1,也就是说从最右边第一个1开始的位往后都取反了,和它自身做“与”运算后改变了的位(从右边第一个1开始往后的位)都会是0,其余的不变。举个例子看一下会更直观,1100,先减去1得到1011,1011 & 1100 = 1000;第二次运算:1000 - 1 = 0111; 0111 & 1000 = 0;循环完毕,共执行了2次。

看来解决问题不是终点,终点是找到最优解决方案。继续努力,祝我拿到一个好的offer吧~

你可能感兴趣的:(算法,位操作)