问题描述:对于一个字节的无符号整型变量,求其二进制表示中“1”的个数,要求算法的执行效率尽可能快。
解法一:
我们知道如果一个数除以2,就相当于这个数右移1位,例如1001 0010:
第一次除以2时,商为0100 1001 ,余数为0;
第二次除以2时,商为0010 0100,余数为1;
因此我们可以通过相除和判断余数来求解,代码如下:
int count(unsigned char n) { int num = 0; while (n) { if (n % 2 == 1) num++; n /= 2; } return num; }
解法二:使用移位操作
1,否则计算结果为0。我们在向右移位的过程中直接把最后一位丢弃。可以把该数和0x01做“与”操作,如果最后1位为1则计算结果为
实现代码:
int count(unsigned char n) { int num = 0; while (n) { if ((n&0x01) == 1) num++; n >>= 1; } return num; }解法三:
能设计个算法,该算法的时间复杂度为二进制中“1“的个数?例如给定一个二进制数0100 0000,如何进行一次操作就能计算出其中1的个数呢?我们可以让该数与一个数做“与”操作,0100 0000 & 0011 1111 = 0,即0100 0000 &(0100 0000 - 1)=0。虽然移位操作比除、余操作效率高了不少,但是移位操作的时间复杂度任为O(N),N为二进制的位数。我们能不
代码为:
int count(unsigned char n) { int num = 0; while (n) { num++; n &= n - 1; } return num; }解法四:查表法
这是一个典型的空间换时间的做法,把0~255中“1”的个数直接存储在数组中,用要求的数做为下标,算法的时间复杂度为O(1)。
代码如下:
int count(unsigned char n) { int table[256]= { 0,1,1,2,1,2,2,3,1,2,2,3,2,3,3,4,1,2,2,3,2,3,3,4,2,3, 3,4,3,4,4,5,1,2,2,3,2,3,3,4,2,3,3,4,3,4,4,5,2,3,3, 4,3,4,4,5,3,4,4,5,4,5,5,6,1,2,2,3,2,3,3,4,2,3,3,4, 3,4,4,5,2,3,3,4,3,4,4,5,3,4,4,5,4,5,5,6,2,3,3,4,3, 4,4,5,3,4,4,5,4,5,5,6,3,4,4,5,4,5,5,6,4,5,5,6,5,6, 6,7,1,2,2,3,2,3,3,4,2,3,3,4,3,4,4,5,2,3,3,4,3,4,4, 5,3,4,4,5,4,5,5,6,2,3,3,4,3,4,4,5,3,4,4,5,4,5,5,6, 3,4,4,5,4,5,5,6,4,5,5,6,5,6,6,7,2,3,3,4,3,4,4,5,3, 4,4,5,4,5,5,6,3,4,4,5,4,5,5,6,4,5,5,6,5,6,6,7,3,4, 4,5,4,5,5,6,4,5,5,6,5,6,6,7,4,5,5,6,5,6,6,7,5,6,6, 7,6,7,7,8 }; return table[n]; }