位运算符
在看一些源码时,经常出现一些位运算符,在此总结一下
左移( << )、右移( >> ) 、无符号右移( >>> ) 、&(与)、|(或)、~(非 为一元运算符)
^(异或)、<<=左移位赋值、>>=右移位赋值、>>>=无符号右移位赋值
&=按位与赋值、|=按位或赋值、^=按位异或赋值
1.与运算符(&)
两个操作数中,位都为1,结果才为1,否则结果为0
int a=8;
int b=7;
System.out.println(a&b);
//输出为:0
//a=8 转为二进制 0000 0000 0000 0000 0000 0000 0000 1000
//b=7 转为二进制 0000 0000 0000 0000 0000 0000 0000 0111
//与运算结果 0000 0000 0000 0000 0000 0000 0000 0000
2.或运算符(|)
两个位,只要有一个为1,那么结果就是1,否则就为0
int a=8;
int b=7;
System.out.println(a|b);
//输出为:15
//a=8 转为二进制 0000 0000 0000 0000 0000 0000 0000 1000
//b=7 转为二进制 0000 0000 0000 0000 0000 0000 0000 0111
//或运算结果 0000 0000 0000 0000 0000 0000 0000 1111
3.非运算符(~)
如果位为0,结果是1,如果位为1,结果是0;一元运算符
int a=8;
System.out.println(~a);
//输出为:-9
//a=8 转为二进制 0000 0000 0000 0000 0000 0000 0000 1000
//非运算结果 1111 1111 1111 1111 1111 1111 1111 0111 ==>化为十进制-9
//对于负数,高位用1表示,正数高位用0表示
//减1取反,最高位符号位不变 1000 0000 0000 0000 0000 0000 0000 1001 => -9
4.异或运算符(^)
两个操作数的位中,相同则结果为0,不同则结果为1
int a=10;
int b=7;
System.out.println(a^b);
//输出为:13
//a=8 转为二进制 0000 0000 0000 0000 0000 0000 0000 1010
//b=7 转为二进制 0000 0000 0000 0000 0000 0000 0000 0111
//异或运算结果 0000 0000 0000 0000 0000 0000 0000 1101 ===> 化为十进制:13
5.左移( << )
a<<b :将a二进制左移b位
int a=7;
System.out.println(a<<2);
//输出为:28
//a=7 转为二进制 0000 0000 0000 0000 0000 0000 0000 0111
//右移两位: 0000 0000 0000 0000 0000 0000 0001 1100 ==>结果为:28
//快速计算= 7*2^2 a<<b =a*2^b
6.右移( >> )
a>>b:将a二进制右移b位
int a=7;
System.out.println(a>>2);
//输出为:1
//a=7 转为二进制 0000 0000 0000 0000 0000 0000 0000 0111
//右移两位: 0000 0000 0000 0000 0000 0000 0000 0001 (11) ==>结果为:1
7.无符号右移( >>> )
正数右移,高位用0补,负数右移,高位用1补,当负数使用无符号右移时,用0进行补位(自然而然的,就由负数变成了正数了)
int a=7;
int b=-7;
System.out.println(a>>>2);
System.out.println(b>>>2);
//输出:1 和 1073741822
//a=7 转为二进制 0000 0000 0000 0000 0000 0000 0000 0111
//无符号右移两位 0000 0000 0000 0000 0000 0000 0000 0001
//b=-7 转为二进制 1111 1111 1111 1111 1111 1111 1111 1001
//无符号右移两位 0011 1111 1111 1111 1111 1111 1111 1110 ==>2^30-2=1073741822
注意:正数或者负数左移,低位都是用0补。
8.由位运算操作符衍生而来的有:
&= 按位与赋值 如:a&=b <=> a=a&b 以下相同
|= 按位或赋值 、^=按位异或赋值 、<<=左移位赋值、>>=右移位赋值、>>>=无符号右移位赋值
ps:关于原码,反码,补码
在计算机中,
正数是直接用原码表示的,如单字节5,在计算机中就表示为:0000 0101。
负数以其正值的补码形式表示,如单字节-5,在计算机中表示为1111 1011。
源码:
正数原码,绝对值大小转换成的二进制数; 0000 0101
负数原码,绝对值大小转换成的二进制数,然后最高位补1。 1000 0101
反码:
正数的反码与原码相同 0000 0101
负数的反码为对该数的原码除符号位外各位取反 1111 1010
补码:
正数的补码与原码相同; 0000 0101
负数的补码为对该数的原码除符号位外各位取反,然后在最后一位加1。1111 1011
- 正数的反码和补码都与原码相同。
- 而负数的反码为对该数的原码除符号位外各位取反。
- 负数的补码为对该数的原码除符号位外各位取反,然后在最后一位加1