JavaScript中的~运算符

最近看《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),先把数字n1,再把以上的结果乘以-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)~-10,这种情况下为false,也符合预期。到此取交集函数intersect分析完毕。

同时按位非运算符还有一个常用的场景就是对浮点数取整 ,另一篇文章JS中!!和~~的作用分别是什么,有什么区别? 中也有关于~~相关的介绍。

js 复制代码
~~3.1415926 // 3,~3.1415926结果为-4,~-4结果为3,就实现了取整

按位非~进行按位取反的过程中会将浮点数位去掉,只对前面32位整数进行处理,由于~的这个特性,小数点后面的部分是直接被去掉的,而不是进行Math.floor这样的四舍五入操作。

相关推荐
一城烟雨_4 小时前
vue3 实现将html内容导出为图片、pdf和word
前端·javascript·vue.js·pdf
一纸忘忧6 小时前
成立一周年!开源的本土化中文文档知识库
前端·javascript·github
软件技术NINI8 小时前
html css js网页制作成品——HTML+CSS甜品店网页设计(4页)附源码
javascript·css·html
涵信8 小时前
第十一节:性能优化高频题-响应式数据深度监听问题
javascript·vue.js·性能优化
codingandsleeping8 小时前
Express入门
javascript·后端·node.js
Vaclee8 小时前
JavaScript-基础语法
开发语言·javascript·ecmascript
拉不动的猪9 小时前
前端常见数组分析
前端·javascript·面试
一个专注写代码的程序媛10 小时前
vue组件间通信
前端·javascript·vue.js
一笑code10 小时前
美团社招一面
前端·javascript·vue.js
懒懒是个程序员11 小时前
layui时间范围
前端·javascript·layui