java位运算详解

概述

  日常开发中位运算不是很常用,但是巧妙的使用位运算可以大量减少运行开销,优化算法。在很多基础类、框架中运用很广泛(HashMap),每当看源码时总困惑经过位运算后到底是什么值。

数字基础概念

  先了解计算机表示数值的几个基础概念:机器数,真值,原码,反码,补码

1、机器数

  在计算机中数据以二进制的形式存在的,而一个数值在计算机中的二进制表示形式,就是这个数的机器数。机器数是有符号位的,在计算机中用一个二进制数的最高位存放符号,正数为0,负数为1,如下实例(按原码表示):
十进制的+7,计算机字长为8位,其二进制就是00000111
十进制的 -7,计算机字长为8位,其二进制就是10000111(这里用的是原码)
其中00000111和10000111就是机器数

2、真值

  由于机器数的第一位是符号位,所以其形式值就不等于其真值的数值,也就是说10000111表示的是-7而不是135(10000111的十进制是135,前提是不算最高位为符号位),因此-7才是机器数的真值。

3、原码

  原码是一种计算机中对数字的二进制定点表示方法。原码表示法在数值前面增加了一位符号位(即最高位为符号位):正数该位为0,负数该位为1,其余位表示数值的大小。
[+7]=[00000111](原码)
[ - 7]=[10000111](原码)
因为第一位是符号位,因此8位二进制的取值范围就是[1111 1111,0111 1111]也就是[-127,127]

4、反码

反码是表示数值的一种形式
正数的反码是其本身
负数的反码是在其原码的基础上, 符号位不变,其余各个位取反.
[+7]=[00000111](原码)= [00000111](反码)
[ - 7]=[10000111](原码)= [11111000](反码)

5、补码

  在计算机系统中,数值一律用补码来表示和存储。原因在于,使用补码,可以将符号位和数值域统一处理;同时,加法和减法也可以统一处理。补码的表示方法是:
正数的补码就是其本身
负数的补码是在其原码的基础上, 符号位不变, 其余各位取反, 最后+1. (即在反码的基础上+1)
[+7]=[00000111](原码)= [00000111](反码)= [00000111](补码)
[ - 7]=[10000111](原码)= [11111000](反码)= [11111001](补码)

位运算符

运算符意义
&按位与
按位或
~按位非
^按位异或
<<左移
>>右移
>>>无符号右移

1、按位与&

  对应位同为1时,才为1,否则全为0(对应位只要有0,全为0,否则为1)。

操作十进制二进制
操作数130011
操作数250101
&结果10001

2、按位或 |

  对应位只要有1时,即为1,否则全为0(对应位只有全是0时,结果才是0,否则为1)。

操作十进制二进制
操作数130011
操作数250101
∣结果70111

3、按位非~

  对每位进行取反。

操作十进制二进制
操作数130011
~结果121100

4、按位异或 ^

  只要对应为不同即为1

操作十进制二进制
操作数130011
操作数250101
^结果60110

5、左移<<

m<<n即在数字没有溢出的前提下,对于正数和负数,左移n位都相当于m乘以2的n次方。
溢出情况举例:5<<29,相当于 (2^2 + 1) * 2^29 ,这个结果显然大于正数的最大值 2^31-1,所以得出的是个负数。
在这里插入图片描述

6、右移>>

m>>n即相当于m除以2的n次方,得到的为整数时,即为结果。如果结果为小数,此时会出现两种情况:
如果m为正数,得到的商会无条件 的舍弃小数位;
如果m为负数,舍弃小数部分,然后把整数部分加+1得到位移后的值。

在这里插入图片描述

7、无符号右移 >>>

  无符号右移>>> 与 右移>> 的区别就是无论操作数是正数还是负数,高位都是补0。

总结

  位运算的优势是速度快,因为位运算直接运算的是计算机底层的二进制机器操作指令。所以在很多框架中使用。例如HashMap里面的hash算法,而不是hashCode() % length
jdk1.7:

static final int indexFor(int h,int length) {
    return (h >> length-1);
}

jdk1.8:扰动函数,为了hash冲突更少
https://www.cnblogs.com/eycuii/p/12015283.html
https://www.zhihu.com/question/20733617

static final int hash(Object key) {
    int h;
    return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16);
}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值