搜索
bottom↓
回复: 13

精妙的算法3,4个循环就可以计算8位二进制中1的个数

[复制链接]

出0入0汤圆

发表于 2022-8-16 22:38:13 | 显示全部楼层 |阅读模式
本帖最后由 cnxh 于 2022-8-16 22:40 编辑

网上找的

如何计算二进制中1的个数?

全部遍历一遍?

不……不……这不是最优解……

下面看一段代码:

int bitcount (unsigned int n)
{
    int count=0 ;
    while (n) {
        count++ ;
        n &= (n - 1) ;
    }
    return count ;
}

太巧啦~

刚看到这个代码的时候,觉得这个解法真是太巧啦、太精妙啦(不要怪我没见过世面 )~

下面我们来理解一下这个代码,这个代码中核心的代码只有一行,就是 n &= (n - 1) ,我们分开看一下:
1. n-1:一个二进制的数减1,就是将这个二进制最右边的那个1变成0,然后它后边的所有位置都变成1~ 举例:0011 0100,减1(n-1)后变成:0011 0011。
2. n &= (n-1),并操作就会将有0的位置都变成0,上面的例子的结果就是0011 0000,然后再赋值给n,这时n就去掉了一个1,或者叫做计数了一个1。以此类推,就可以得到1的个数。

0011 0100 - 1 = 0011 0011   
0011 0100 & 0011 0011 = 0011 0000  // 计数一个1
0011 0000 - 1 = 0010 1111
0011 0000 & 0010 1111 = 0010 0000  // 计数两个1
0010 0000 - 1 = 0001 1111
0010 0000 & 0001 1111 = 0000 0000  // 计数三个1,程序停止

这样只需要循环3次,而如果直接遍历就会循环6次,性能明显提高啦~

给个极端的例子 1000 0000,这个直接遍历得遍历8次,而通过上面的代码只需要遍历

1次~1次~~1次~~~

真是太精妙啦~

出0入0汤圆

 楼主| 发表于 2022-8-18 07:17:55 | 显示全部楼层
1a2b3c 发表于 2022-8-17 01:44
比如说,在一些资源或者类似的缺乏的地方,用每个bit表示有多少个状态,例如一个字节的8bit代表了8个故障 ...
(引用自8楼)

现在用串口模拟发送,需要校验位计算
回帖提示: 反政府言论将被立即封锁ID 在按“提交”前,请自问一下:我这样表达会给举报吗,会给自己惹麻烦吗? 另外:尽量不要使用Mark、顶等没有意义的回复。不得大量使用大字体和彩色字。【本论坛不允许直接上传手机拍摄图片,浪费大家下载带宽和论坛服务器空间,请压缩后(图片小于1兆)才上传。压缩方法可以在微信里面发给自己(不要勾选“原图),然后下载,就能得到压缩后的图片】。另外,手机版只能上传图片,要上传附件需要切换到电脑版(不需要使用电脑,手机上切换到电脑版就行,页面底部)。
您需要登录后才可以回帖 登录 | 注册

本版积分规则

手机版|Archiver|amobbs.com 阿莫电子技术论坛 ( 粤ICP备2022115958号, 版权所有:东莞阿莫电子贸易商行 创办于2004年 (公安交互式论坛备案:44190002001997 ) )

GMT+8, 2024-5-11 14:12

© Since 2004 www.amobbs.com, 原www.ourdev.cn, 原www.ouravr.com

快速回复 返回顶部 返回列表