求二进制中1的个数
2016-05-13 10:03
190 查看
问题描述:对于一个字节的无符号整型变量,求其二进制表示中“1”的个数,要求算法的执行效率尽可能快。
解法一:
我们知道如果一个数除以2,就相当于这个数右移1位,例如1001 0010:
第一次除以2时,商为0100 1001 ,余数为0;
第二次除以2时,商为0010 0100,余数为1;
因此我们可以通过相除和判断余数来求解,代码如下:
解法二:使用移位操作
我们在向右移位的过程中直接把最后一位丢弃。可以把该数和0x01做“与”操作,如果最后1位为1则计算结果为
1,否则计算结果为0。
实现代码:
虽然移位操作比除、余操作效率高了不少,但是移位操作的时间复杂度任为O(N),N为二进制的位数。我们能不
能设计个算法,该算法的时间复杂度为二进制中“1“的个数?例如给定一个二进制数0100 0000,如何进行一次操作就能计算出其中1的个数呢?我们可以让该数与一个数做“与”操作,0100 0000 & 0011 1111 = 0,即0100 0000 &(0100 0000 - 1)=0。
代码为:
这是一个典型的空间换时间的做法,把0~255中“1”的个数直接存储在数组中,用要求的数做为下标,算法的时间复杂度为O(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; }
解法二:使用移位操作
我们在向右移位的过程中直接把最后一位丢弃。可以把该数和0x01做“与”操作,如果最后1位为1则计算结果为
1,否则计算结果为0。
实现代码:
int count(unsigned char n) { int num = 0; while (n) { if ((n&0x01) == 1) num++; n >>= 1; } return num; }解法三:
虽然移位操作比除、余操作效率高了不少,但是移位操作的时间复杂度任为O(N),N为二进制的位数。我们能不
能设计个算法,该算法的时间复杂度为二进制中“1“的个数?例如给定一个二进制数0100 0000,如何进行一次操作就能计算出其中1的个数呢?我们可以让该数与一个数做“与”操作,0100 0000 & 0011 1111 = 0,即0100 0000 &(0100 0000 - 1)=0。
代码为:
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 ; }
相关文章推荐
- 数据结构(六)——二叉树 前序、中序、后序、层次遍历及非递归实现 查找、统计个数、比较、求深度的递归实现
- codeforces 672D Robin Hood (二分)
- 【C++】int与string的互转
- Linux php 中文乱码的快速解决方法
- spring+hibernate:在applicationCOntext.XML中配置C3P0参数说明
- Vim新手节省时间的10多个小技巧
- LeetCode 22 Generate Parentheses
- 迈克菲:2016年的八大网络安全威胁
- tomcat+nginx+redis实现均衡负载、session共享(一)
- MFC 中解析一个路径
- Django - “No migrations to apply” when run migrate after makemigrations
- jquery 的滚动条插件 jquery.nicescroll.js
- 关于JSONP
- visual studio下CUDA7.5的安装
- Tomcat指定JAVA_HOME而不用默认环境变量
- Objective-C编码规范:26个方面解决iOS开发问题
- UIMenuController和UIMenuItem,即iOS剪贴板
- 如何在批次檔(Batch)中實現 sleep 命令讓任務暫停執行 n 秒
- 无线信号强度解析及linux如何查看wifi信号强弱等
- jstack和线程dump分析