最近看《JavaScript框架设计》这本书的2.4 EXT数组扩展这一节,intersect方法:对两个数组取交集。
js
function intersect(target, array) {
return target.filter(function(n) {
return ~array.indexOf(n);
});
}
第三行的~
一时间没看懂,专程研究了一下。先使用一下上面的函数,如下:
js
function intersect(target, array) {
return target.filter(function(n) {
return ~array.indexOf(n);
});
}
let arr = [1, 2, 3];
let arr1 = [3, 6, 9];
console.log(intersect(arr, arr1)); // [3]
intersect
第三行的~
是按位非运算符。(其他位运算相关的文章 十进制的负数和十六进制间相互转换,ECMAScript中的基本概念-位操作符,JS中!!和~~的作用分别是什么,有什么区别?。)按位非运算符将操作数的位反转。如同其他位运算符一样,它将操作数转化为 32 位的有符号整型。比如数字5
的按位非即是~5
,这是怎么来的呢?~5
是多少呢?看下面的演示。
js
00000000000000000000000000000101 前面是数字5的32位二进制表示
11111111111111111111111111111010 对上面5的二进制表示按位取反,
这里的值为-6,这是怎么来的呢?
在ECMAScript中的基本概念-位操作符这里已经介绍过,负数同样以二进制码存储,但是使用二进制补码。计算一个数值的二进制补码步骤是这样的:
- 先求这个负数的绝对值的二进制码
- 在求以上二进制码的反码
- 对以上得到的二进制反码加1
以-6为例,先把-6的绝对值,数字6用32位二进制表示,再对6的二进制按位取反,再对上一步的反码加1,如下:
js
00000000000000000000000000000110 6的32位二进制表示
11111111111111111111111111111001 对6的二进制按位取反
11111111111111111111111111111010 对上一步的反码加1
可以看到这里的反码加1过后,同~5的二进制表示都是11111111111111111111111111111010
,就可以得出~5 === -6
。再拿一个负数-3
的按位非~-3
来推导一下。
js
00000000000000000000000000000011 -3的绝对值3的二进制表示
11111111111111111111111111111100 对3的二进制按位取反
11111111111111111111111111111101 对上一步的反码加1,前面就是-3的二进制表示
00000000000000000000000000000010 对上面的二进制表示按位取反,一眼就能看出是2
可以得出~-3 === 2
,结合~5 === -6
可以归纳出一个公式,对于数字n
的按位非~n
,~n === -(n + 1)
,先把数字n
加1
,再把以上的结果乘以-1
,或~n === -n - 1
,先把n
乘以-1
,再-1
。
最后回到上面的取交集函数intersect
,遍历target
中的项n
, 如果array
中也有这个遍历的子项n
,那么array.indexOf(n)
应该是大于等于零的,对于一个大于等于零的数取反~array.indexOf(n)
的结果应该是小于零的,这种情况下为true
,符合预期。如果array
中没有这个遍历的子项n
,那么array.indexOf(n)
是等于-1
的,对其取反~array.indexOf(n)
即~-1
为0
,这种情况下为false
,也符合预期。到此取交集函数intersect
分析完毕。
同时按位非运算符还有一个常用的场景就是对浮点数取整 ,另一篇文章JS中!!和~~的作用分别是什么,有什么区别? 中也有关于~~
相关的介绍。
js
~~3.1415926 // 3,~3.1415926结果为-4,~-4结果为3,就实现了取整
按位非~
进行按位取反的过程中会将浮点数位去掉,只对前面32位整数进行处理,由于~
的这个特性,小数点后面的部分是直接被去掉的,而不是进行Math.floor这样的四舍五入操作。