JS神奇的或0(|0)

js · 浏览次数 : 0

小编点评

在JavaScript中,当我们使用按位或运算符`|`与0时,实际上是将数字转换为32位无符号整数。这是因为默认情况下,JavaScript中的数字是基于64位浮点数表示的,但在进行按位运算时,可能会因为精度问题而导致错误。 例如,当我们尝试对一个大于`Number.MAX_SAFE_INTEGER`(9007199254740991)的值进行按位或运算时,可能会得到一个错误的结果,因为它不能被表示为一个32位无符号整数。 然而,在某些情况下,我们可以通过将数字转换为字符串,然后使用字符串拼接的方式来实现按位或运算。例如: ```javascript const num = 9007199254740991; const maxSafeInteger = Number.MAX_SAFE_INTEGER; const result = String(num) + '' | 0; console.log(result); // 输出:-19007199254740990 ``` 在这个例子中,我们将数字转换为字符串,然后使用字符串拼接的方式实现了按位或运算。这种方法可以在一定程度上解决精度问题,但仍然存在一定的风险,因为字符串拼接可能会导致额外的性能开销。 总之,在JavaScript中,为了避免按位或运算中出现错误的结果,我们应该尽量避免对大于`Number.MAX_SAFE_INTEGER`的值进行按位或运算。如果需要进行按位运算,可以考虑使用其他方法,如字符串拼接或自定义函数来处理。

正文

按照常识,位运算x|0,要么等于x,要么等于0
那么在JS的世界你的认知就要被颠覆了
下面请看

不带或0运算:
(window.crypto.getRandomValues(new Uint32Array(1))[0] * 0x10000 )
168546249998336
(window.crypto.getRandomValues(new Uint32Array(1))[0] * 0x10000 )
18707488702464
(window.crypto.getRandomValues(new Uint32Array(1))[0] * 0x10000 )
15579009253376
(window.crypto.getRandomValues(new Uint32Array(1))[0] * 0x10000 )
194841754140672
(window.crypto.getRandomValues(new Uint32Array(1))[0] * 0x10000 )
262611854950400
(window.crypto.getRandomValues(new Uint32Array(1))[0] * 0x10000 )
171394313420800

带或0运算:
(window.crypto.getRandomValues(new Uint32Array(1))[0] * 0x10000 )|0
-1037238272
(window.crypto.getRandomValues(new Uint32Array(1))[0] * 0x10000 )|0
511180800
(window.crypto.getRandomValues(new Uint32Array(1))[0] * 0x10000 )|0
1204224000
(window.crypto.getRandomValues(new Uint32Array(1))[0] * 0x10000 )|0
-2026438656

可以看到明显的带或0运算与不带或0运算的结果无论是位数还是符号位都有不同。
那这中间到底发生了什么?

这里找一个数字为例:117063531626496
要想验证这个问题,思路如下:
1,对比变更前后的数字的二进制格式
2,找到是否有数字表示的安全边界

首先按照思路1,我们看一下这个数字和这个数字或0后的二进制格式分别是什么:

117063531626496的二进制格式:

var num = 117063531626496; num.toString(2);
输出:'11010100111011111111010001110000000000000000000'

117063531626496 | 0
输出:-96993280

-96993280的二进制格式:
var num = -96993280; num.toString(2);
'-101110010000000000000000000'

对比对比:

11010100111011111111010001110000000000000000000
                   -101110010000000000000000000

除了后面的0位数相同,没有找到明显的线索

那么我们按照思路2,来看一下原因:
通过官网对于js的number的定义,是64位的统一类型
但是我们通过 Number.MAX_SAFE_INTEGER可以看到number的安全最大值是:9007199254740991

通过转为2进制,可以发现这个数字是个54位的1:
var num = 9007199254740991; num.toString(2)
'11111111111111111111111111111111111111111111111111111'

那这个值可以正常地或0吗?实际上还是不行

9007199254740991|0
-1
9007199254740990|0
-2

那这个边界到底是多少呢?对于其它语言Integer的默认最大值一般为2的32次方-1,也就是2147483647
这次再来试一下:

2147483647|0
2147483647
如果对这个值再+1,重试呢
2147483648|0
-2147483648

可以发现这个边界就是32位整数的最大值。
超过这个值的数字再与0进行位运算则可能得到一个错误的结果。

与JS神奇的或0(|0)相似的内容:

JS神奇的或0(|0)

按照常识,位运算x|0,要么等于x,要么等于0 那么在JS的世界你的认知就要被颠覆了 下面请看 不带或0运算: (window.crypto.getRandomValues(new Uint32Array(1))[0] * 0x10000 ) 168546249998336 (window.cryp

JS语法让人困惑的点 “==与===”

在JS中有很多神奇的语法,非常让人困惑,我们就先一一道来,相信你在开发中或多或少都踩过这些坑,或者让人无法理解。 今天我们就来说下 "`==`" 和 "`===`" 这题对于很多没有系统学过前端开发的技术人员来说,算个重点,来画起来,我们一起看。

《爆肝整理》保姆级系列教程-玩转Charles抓包神器教程(8)-Charles如何进行断点调试

1.简介 Charles和Fiddler一样也有个强大的功能,可以修改发送到服务器的数据包,但是修改前需要拦截,即设置断点。设置断点后,开始拦截接下来所有网页,直到取消断点。这个功能可以在数据包发送之前,修改请求参数;在收到应答包,在js解析和浏览器渲染之前,修改返回结果。有了这个功能,开发者就可以

js定时器

一.定时器 1. JS存在两种定时器 setTimeout() 延迟定时器 setInterval() 循环定时器(“间隔器”) 定时器中的函数挂载在window对象,内部的this ——> window setTimerout(function(){ console.log('wuwei') },

调用了这么久的JS方法是长在对象、类、值本身还是原型链上?

调用了这么久的JS方法是长在对象、类、值本身还是原型链上? JavaScript这门语言总是能带给我惊喜,在敲代码的时候习以为常的写法,退一步再看看发现自己其实对很多基操只有表面的使用,而从来没思考过为何要这样操作。 今天整理JS代码的时候突然发出灵魂三连问: 为什么有些时候操作对象,可以直接调用对

什么是浅拷贝和深拷贝,如何用 js 代码实现?

〇、简介和对比 简介 浅拷贝:只复制原始对象的第一层属性值。 如果属性值是值类型,将直接复制值,本值和副本变更互不影响; 如果是引用数据类型,则复制内存地址,因此原始对象和新对象的属性指向相同的内存地址,改变任一值,另一变量值也会同步变更。 深拷贝:递归地复制原始对象的所有层级。 每一个属性值都会在

记录工作中常用的 JS 数组相关操作

工作中难免会遇到各种各样的数据结构,较为全面的了解数组操作,对于复杂数据结构的处理会非常有用且节省时间。所以想在这里总结一下工作中常用的数组操作,都是一些非常基础的知识,大家看个乐就好~

【技巧】JS代码这么写,前端小姐姐都会爱上你

这篇文章分享了JavaScript编程中的实用技巧,包括解构赋值的短路语法避免错误、深度解构及默认值设定,以及数组操作如条件添加元素、获取最后一个元素和使用includes优化条件判断。此外,还介绍了从URL解析参数、页面滚动功能和获取滚动距离的JS片段。作者提倡使用这些技巧提升代码质量和效率,并邀...

JS 监听用户页面访问&页面关闭并进行数据上报操作

用户在页面访问时发送数据到后台,页面关闭时也发送数据到后台。 第一次进入页面时触发页面访问 刷新当前页面时触发页面访问 新 tab 进入页面时触发页面访问 当前页面点击 nav 进入其他模块时,触发页面关闭&页面访问 关闭页面时触发页面关闭

JS 实现鼠标框选(页面选择)时返回对应的代码或文本内容

当用户进行鼠标框选选择了页面上的内容时,把选择的内容进行上报。 分为以下几点: 选择文案时 选择图片、svg、iframe、video、audio 等标签时 选择 input、select、textarea 等标签时 选择input、textarea 标签内容时 选择类似   字符时 键盘全选时 鼠...