1.二进制或运算符(or):符号为 |
逐位比较两个运算子,两个二进制位之中只要有一个为1,就返回1,否则返回0。
0 | 3 // 3
上面代码中,0和3的二进制形式分别是00和11,所以进行二进制或运算会得到11(即3)。
注意
:位运算只对整数有效,遇到小数时,会将小数部分舍去,只保留整数部分,并且这种取整方法不 适用超过32位整数最大值2147483647的数。
2.二进制与运算符(and):符号为&
逐位比较两个运算子,两个二进制位之中只要有一个位为0,就返回0,否则返回1。
0 & 3 // 0
上面代码中,0(二进制00)和3(二进制11)进行二进制与运算会得到00(即0)。
3.二进制否运算符(not):符号为~
将每个二进制位都变为相反值(0变为1,1变为0)
~ 3 // -4
大家一定很疑惑上面的结果怎么得来的呢?因为JavaScriptt 内部将所有的运算子都转为32位的二进制整数再进行运算。
运算过程如下:
3的32位整数形式是00000000000000000000000000000011,二进制否运算以后得到11111111111111111111111111111100。由于第一位(符号位)是1,所以这个数是一个负数。JavaScript
内部采用补码形式表示负数,即需要将这个数减去1,再取一次反,然后加上负号,才能得到这个负数对应的10进制值。这个数减去1等于11111111111111111111111111111011,再取一次反得到00000000000000000000000000000100,再加上负号就是-4。
考虑到这样的过程比较麻烦,可以简单记忆成,一个数与自身的取反值相加,等于-1
即3取反等于-4,3与-4相加得-1
~ -3 // 2
4.异或运算符(xor):符号为^
两个二进制位不同时返回1,相同时返回0。
0 ^ 3 // 3
上面表达式中,0(二进制00)与3(二进制11)进行异或运算,它们每一个二进制位都不同,所以得到11(即3)。
5.左移运算符(left shift):符号为<<
将一个数的二进制值向左移动指定的位数,尾部补0
,即乘以2的指定次方。向左移动的时候,最高位的符号位是一起移动的。
// 4 的二进制形式为100,
// 左移一位为1000(即十进制的8)
// 相当于乘以2的1次方
4 << 1
// 8
-4 << 1
// -8
上面代码中,-4左移一位得到-8,是因为-4的二进制形式是11111111111111111111111111111100,左移一位后得到11111111111111111111111111111000,该数转为十进制(减去1后取反,再加上负号)即为-8。
6.右移运算符(right shift):符号为>>
将一个数的二进制值向右移动指定的位数
。如果是正数,头部全部补0;如果是负数,头部全部补1。右移运算符基本上相当于除以2的指定次方(最高位即符号位参与移动)。
4 >> 1 // 2
/*
// 因为4的二进制形式为 00000000000000000000000000000100,
// 右移一位得到 00000000000000000000000000000010,
// 即为十进制的2
*/
-4 >> 1 // -2
/*
// 因为-4的二进制形式为 11111111111111111111111111111100,
// 右移一位,头部补1,得到 11111111111111111111111111111110,
// 即为十进制的-2
*/
为了便于计算左移运算符相当于乘2,右移运算符相当于除2
7.头部补零的右移运算符(zero filled right shift):符号为>>>
头部补零的右移运算符(>>>)与右移运算符(>>)只有一个差别
,就是一个数的二进制形式向右移动时,头部一律补零,而不考虑符号位
.。所以,该运算总是得到正值。对于正数,该运算的结果与右移运算符(>>)完全一致,区别主要在于负数。
4 >>> 1
// 2
-4 >>> 1 // 2147483646
/*
// 因为-4的二进制形式为11111111111111111111111111111100,
// 带符号位的右移一位,得到01111111111111111111111111111110,
// 即为十进制的2147483646。
*/