剖析大神代码,计算整型里面1的个数
发布日期:2021-06-30 18:44:43 浏览次数:2 分类:技术文章

本文共 2565 字,大约阅读时间需要 8 分钟。

昨天的文章,可能很多人看了不知道怎么回事,确实,我也是看了之后一头雾水。

先给出个简单的例子

#include "stdio.h"int count_bits4(char x){	x = (x&0x55) + ((x>>1)&0x55);	x = (x&0x33) + ((x>>2)&0x33);	x = (x&0x0F) + ((x>>4)&0x0F);	return x;}int main(){	printf("%x\n",count_bits4(0x12));}

输出:

我们现看看下面这段 0x55 = 0b01010101 0x33 = 0b00110011 0x0F = 0b00001111

这个图片需要解释一下 相邻两个数不断相加,直到这个数据的长度

#举个例子

比如我有一个数字,0x34。我要计算0x34里面有多少个1。转成二进制是 0b00110100,这时候,我把它们相邻的二进制位先相加起来。第二步,就是需要把相邻两位的数加起来,第三步,就是把相邻的四位数加起来。

#复杂一点的

#include "stdio.h"int count_bits3(long long s){    s = (s&0x5555555555555555L) + ((s>>1)&0x5555555555555555L);    s = (s&0x3333333333333333L) + ((s>>2)&0x3333333333333333L);    s = (s&0x0F0F0F0F0F0F0F0FL) + ((s>>4)&0x0F0F0F0F0F0F0F0FL);    s = (s&0x00FF00FF00FF00FFL) + ((s>>8)&0x00FF00FF00FF00FFL);    s = (s&0x0000FFFF0000FFFFL) + ((s>>16)&0x0000FFFF0000FFFFL);    s = (s&0x00000000FFFFFFFFL) + ((s>>32)&0x00000000FFFFFFFFL);    return (int)s;}int main(){	printf("%x\n",count_bits3(0x122312351345134));}

输出:

我们搞清楚了上面的那段代码,再看下面这段代码,是不是觉得没那么吃力了?

#实测大神的代码如何吊打普通代码

test1.c

#include "stdio.h"#include 
int count_bits(int x){  register int xx=x;  xx=xx-((xx>>1)&0x55555555);  xx=(xx&0x33333333)+((xx>>2)&0x33333333);  xx=(xx+(xx>>4))&0x0f0f0f0f;  xx=xx+(xx>>8);  return (xx+(xx>>16)) & 0xff;}int main(void){    clock_t start, end;    start = clock();    unsigned int i = 0;    for(i = 0;i<10000000;i++){        count_bits(12345678);    }    printf("%d\n",count_bits(12345678));    end = clock();    double seconds  =(double)(end - start)/CLOCKS_PER_SEC;    fprintf(stderr, "Use time is: %.8f s\n", seconds);    return (0);}

test2.c

#include "stdio.h"#include 
int count_bits2(int x){    int i = 0;    int s = 0;    for(i = 0;i<32;i++)    {        if(x&0x01)            s ++;        x = x>>1;    }    return (s);}int main(void){    clock_t start, end;    start = clock();    unsigned int i = 0;    for(i = 0;i<10000000;i++){        count_bits2(12345678);    }    printf("%d\n",count_bits2(12345678));    end = clock();    double seconds  =(double)(end - start)/CLOCKS_PER_SEC;    fprintf(stderr, "Use time is: %.8f s\n", seconds);    return (0);}

运行:

$ gcc test2.c -o test2$ gcc test1.c -o test1$ ./test112Use time is: 0.02542700 s./test212Use time is: 0.44449600 s$

#其他方法

在上一篇文章里面,有几个读者说了另外几种方法用来解决这个问题的代码。欣赏一下

代码是程序员的艺术,认真对待代码,让自己变成一个和亚里士多德一样优秀的工程师。

u8 cnt;while(value){    value &= value - 1;    cnt++;}return cnt;
public int NumberOf1(int n) {    int count = 0;    while (n != 0) {        n = n & (n - 1);        count += 1;    }    return count;}

  回复「 篮球的大肚子」进入技术群聊

回复「1024」获取1000G学习资料

转载地址:https://linus.blog.csdn.net/article/details/105108594 如侵犯您的版权,请留言回复原文章的地址,我们会给您删除此文章,给您带来不便请您谅解!

上一篇:Linux内核0.12完全注释
下一篇:看看大神是如何计算32位数中‘1’的个数

发表评论

最新留言

感谢大佬
[***.8.128.20]2024年04月13日 01时13分27秒

关于作者

    喝酒易醉,品茶养心,人生如梦,品茶悟道,何以解忧?唯有杜康!
-- 愿君每日到此一游!

推荐文章

Image Pyramid(图像金字塔) 2019-04-30
Oracle 作业记录 2019-04-30
putty连接AWS配置(multimedia project) 2019-04-30
Hourglass Network 沙漏网络 (pose estimation姿态估计) 2019-04-30
OpenCV实战(二)——答题卡识别判卷 2019-04-30
目标检测神经网络的发展历程(52 个目标检测模型) 2019-04-30
Boundary loss 损失函数 2019-04-30
神经网络调参实战(一)—— 训练更多次数 & tensorboard & finetune 2019-04-30
tensorflow使用tensorboard进行可视化 2019-04-30
神经网络调参实战(二)—— activation & initializer & optimizer 2019-04-30
凸优化 convex optimization 2019-04-30
数据库索引 & 为什么要对数据库建立索引 / 数据库建立索引为什么会加快查询速度 2019-04-30
IEEE与APA引用格式 2019-04-30
research gap 2019-04-30
pytorch训练cifar10数据集查看各个种类图片的准确率 2019-04-30
Python鼠标点击图片,获取点击点的像素坐标 2019-04-30
路径规划(一) —— 环境描述(Grid Map & Feature Map) & 全局路径规划(最优路径规划(Dijkstra&A*star) & 概率路径规划(PRM&RRT)) 2019-04-30
神经网络调参实战(四)—— 加深网络层次 & 批归一化 batch normalization 2019-04-30
数据挖掘与数据分析(三)—— 探索性数据分析EDA(多因子与复合分析) & 可视化(1)—— 假设检验(μ&卡方检验&方差检验(F检验))&相关系数(皮尔逊&斯皮尔曼) 2019-04-30
RRT算法(快速拓展随机树)的Python实现 2019-04-30