探索Integer番外篇之Integer.bitCount(int i)

1. 前言

钱不是真正的花掉了,而是换了一种方式陪在你身边

bitCount:统计int类型数值的二进制中1的个数。
在各种版本的面试宝典中,这个题目应该是非常常见的了。比如《剑指offer》面试题10:二进制中1的个数,《编程之美》2.1 求二进制数中1的个数,实现方法非常之多,各有千秋。
我们来看看jdk中是如何实现的。

2. 源码


相比较其他的解法,这个是非常新颖的方式,不愧于大师之笔。

    public static int bitCount(int i) {
        // HD, Figure 5-2
        i = i - ((i >>> 1) & 0x55555555);//第1步
        i = (i & 0x33333333) + ((i >>> 2) & 0x33333333);//第2步
        i = (i + (i >>> 4)) & 0x0f0f0f0f;//第3步
        i = i + (i >>> 8);//第4步
        i = i + (i >>> 16);//第5步
        return i & 0x3f;//第6步
    }

解析过程:
输入 i,
i = b_0*2^0 + b_1*2^1 + b_2*2^2 + ... + b_{30}*2^{30} + b_{31}*2^{31} \tag{2.1},
其中 b_0,b_1,b_2...b_{31}
要么为0,要么为1,则,统计二进制1的个数,实则求
b_0+b_1+b_2+...+b_{31}
第1步:i = i - ((i >>> 1) & 0x55555555);
i >>> 1的结果
b_1*2^0 + b_2*2^1 + ... + b_{31}*2^{30}\tag{2.2}
而0x55555555中 5 表示 0101,与运算就是取奇数位(取第0位,第2位,第4位...),则
(i>>>1) & 0x55555555的结果为
b_1*2^0 + b_3*2^2 + b_5*2^4 +...+ b_{31}*2^{30} \tag{2.3}
i = i - ((i >>> 1) & 0x55555555)的结果为(2.2+2.3)
\begin{eqnarray*} i &=& (b_0-b_1)*2^0 + b_1*2^1 + (b_2-b_3)*2^2 + ... + (b_{30}-b_{31})*2^{30} + b_{31}*2^{31} \\&=&(b_0+b_1)*2^0 + (b_2+b_3)*2^2+...+(b_{30}+b_{31})*2^{30} \tag{2.4} \end{eqnarray*}
第2步:i = (i & 0x33333333) + ((i >>> 2) & 0x33333333);
0x33333333中 表示 0011,与运算就是每四位取前两位
i & 0x33333333的结果为
(b_0+b_1) * 2^0 + (b_4+b_5) * 2^4 + (b_8+b_9) * 2^8 +...+(b_{28}+b_{29})*2^{28} \tag{2.5}
(i>>>2) & 0x33333333的结果为
(b_2+b_3) * 2^0 + (b_6+b_7)*2^4 + (b_10+b_11)*2^{8} +...+(b_{30}+b_{31})*2^{28} \tag{2.6}
i = (i & 0x33333333) + ((i>>>2) & 0x33333333)的结果为(2.5 + 2.6)
\begin{eqnarray*} i &=&(b_0+b_1+b_2+b_3)*2^0 + (b_4+b_5+b_6+b_7)*2^4+...+(b_{28}+b_{29}+b_{30}+b_{31})*2^{28} \tag{2.7} \end{eqnarray*}
第3步:i = (i + (i >>> 4)) & 0x0f0f0f0f;
i + (i>>>4)的结果为
\begin{eqnarray*} (b_0+b_1+b_2+b_3+b_4+b_5+b_6+b_7)*2^0 + (b_4+b_5+b_6+b_7+b_8+b_9+b_{10}+b_{11})*2^4 +\\...+(b_{24}+b_{25}+b_{26}+b_{27}+b_{28}+b_{29}+b_{30}+b_{31})*2^{24}+(b_{28}+b_{29}+b_{30}+b_{31})*2^{28} \end{eqnarray*}
而0x0f0f0f0f中,与运算表示每8位取前四位,
则i = (i + (i>>>4)) & 0x0f0f0f0f的结果为
\begin{eqnarray*} i=(b_0+b_1+b_2+b_3+b_4+b_5+b_6+b_7)*2^0 + (b_8+b_9+b_{10}+b_{11} +b_{12}+b_{13}+b_{14}+b_{15})*2^8 +\\...+(b_{24}+b_{25}+b_{26}+b_{27}+b_{28}+b_{29}+b_{30}+b_{31})*2^{24} \end{eqnarray*}
第4步:i = i + (i >>> 8);
i = i + (i>>>8)的结果为
\begin{eqnarray*} i=&(&b_0+b_1+b_2+...+b_{14}+b_{15})*2^0 +(b_8+b_9+b_{10}+...+b_{22}+b_{23})*2^8+ \\&(&b_{24}+b_{25}+b_{26}+...+b_{30}+b_{31})*2^{16} +(b_{24}+b_{25}+b_{26}+...+b_{30}+b_{31})*2^{24} \end{eqnarray*}
第5步:i = i + (i >>> 16);
i = i + (i>>>16)的结果为
\begin{eqnarray*} i=&(&b_0+b_1+b_2+...+b_{30}+b_{31})*2^0 +(b_8+b_9+b_{10}+...+b_{30}+b_{31})*2^8+ \\&(&b_{24}+b_{25}+b_{26}+...+b_{30}+b_{31})*2^{16} +(b_{24}+b_{25}+b_{26}+...+b_{30}+b_{31})*2^{24} \end{eqnarray*}
第5步:i & 0x3f
0x3f表示只取二进制前6位
结果为:
\begin{eqnarray*} i&=&(b_0+b_1+b_2+...+b_{30}+b_{31})*2^0 \\&=&b_0+b_1+b_2+...+b_{30}+b_{31} \end{eqnarray*}
即为二进制1的个数

3.后言

6步法求二进制1的个数,你学会了吗?


其他

本人也是在慢慢学习中,如有错误还请原谅、敬请指出,谢谢!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,772评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,458评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,610评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,640评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,657评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,590评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,962评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,631评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,870评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,611评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,704评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,386评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,969评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,944评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,179评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,742评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,440评论 2 342

推荐阅读更多精彩内容

  • 第2章 基本语法 2.1 概述 基本句法和变量 语句 JavaScript程序的执行单位为行(line),也就是一...
    悟名先生阅读 4,114评论 0 13
  • 在C语言中,五种基本数据类型存储空间长度的排列顺序是: A)char B)char=int<=float C)ch...
    夏天再来阅读 3,320评论 0 2
  • Integer类为java基本类型int的包装类,除了前面提到的Byte类,Short类中的大部分方法,Integ...
    Kinsanity阅读 894评论 0 2
  • 放假前已作了假期规划,带回好几本书,立志暑假要在家看书,不能让时间虚度。 谁知到家没有歇脚的余地,完成一件...
    宁_5687阅读 280评论 0 2
  • 李晨是我的同学。每次他在上课的时候睡觉,我就坐在他旁边,写关于他的故事。下课了,他醒了,我也就写完了。 她脱下上衣...
    晓平阅读 473评论 0 2