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这样的四舍五入操作。

相关推荐
子兮曰13 分钟前
DeepSeek TUI:原生 Rust 打造的终端 AI 编码 Agent
前端·javascript·后端
暗不需求22 分钟前
# 深入 React Todos:从零实现一个状态提升与本地持久化的待办应用
javascript·react.js·全栈
子兮曰29 分钟前
深入 Superpowers:180k Stars 的开源 AI 编程方法论是如何工作的
前端·javascript·后端
隔壁的大叔1 小时前
Markdown 渲染如何穿插自定义组件
前端·javascript·vue.js
薯老板1 小时前
JavaScript原型,原型链
javascript
愚者Pro2 小时前
Flutter基础学习
前端·javascript·vue.js
时光足迹2 小时前
Tiptap 简单编辑器模版
前端·javascript·react.js
吴声子夜歌2 小时前
Vue3——使用Mock.js
javascript·vue·mock.js
时光足迹2 小时前
ThreeJS之GUI控制器
前端·javascript·three.js
时光足迹2 小时前
Tiptap编辑器
前端·javascript·react.js