~~(“双波浪号”)在 Javascript 中做了啥?

Posted

技术标签:

【中文标题】~~(“双波浪号”)在 Javascript 中做了啥?【英文标题】:What does ~~ ("double tilde") do in Javascript?~~(“双波浪号”)在 Javascript 中做了什么? 【发布时间】:2011-05-02 14:55:36 【问题描述】:

我今天在查看一个在线游戏物理库时,偶然发现了 ~~ 运算符。我知道单个 ~ 是按位 NOT,这会使 ~~ 成为 NOT 的 NOT,它会返回相同的值,不是吗?

【问题讨论】:

另见What is the “double tilde” (~~) operator in javascript? 【参考方案1】:

~ 似乎是 -(N+1)。所以~2 == -(2 + 1) == -3 如果你在 -3 上再次执行它,它会返回:~-3 == -(-3 + 1) == 2 它可能只是以一种迂回的方式将字符串转换为数字。

看到这个帖子:http://www.sitepoint.com/forums/showthread.php?t=663275

另外,更详细的信息在这里:http://dreaminginjavascript.wordpress.com/2008/07/04/28/

【讨论】:

感谢 Drackir 的链接!【参考方案2】:

它会删除小数点后的所有内容,因为按位运算符会将其操作数隐式转换为有符号的 32 位整数。无论操作数是(浮点)数字还是字符串,这都有效,结果是数字。

换句话说,它产生:

function(x) 
  if(x < 0) return Math.ceil(x);
  else return Math.floor(x);

仅当 x 介于 -(231) 和 231 - 1 之间。否则,将发生溢出,数字将“环绕”。

这对于将函数的字符串参数转换为数字可能被认为是有用的,但由于可能会溢出,而且它不适合与非整数一起使用,我不会那样使用它,除了“代码高尔夫"(ie 以牺牲可读性和健壮性为代价,毫无意义地从程序的源代码中删除字节)。我会改用+xNumber(x)


这怎么是NOT的NOT

数字-43.2,例如是:

-43.210 = 111111111111111111111111110101012

作为带符号(二进制补码)的 32 位二进制数。 (JavaScript 忽略小数点后的内容。)反转位给出:

非 -4310 = 000000000000000000000000001010102 = 4210

再次反转给出:

非 4210 = 111111111111111111111111110101012 = -4310

这与Math.floor(-43.2) 的不同之处在于负数向零舍入,而不是远离零。 (等于 -44 的 floor 函数总是向下舍入到下一个较小的整数,无论​​该数字是正数还是负数。)

【讨论】:

也就是说,~~ 是创建truncate function 的一种简写方式(也可能是一个好的解决方案?),但显然in javascript。 JSLint 会抱怨使用~~ 试试 Math.trunc()【参考方案3】:

假设~N-(N+1),那么~~N-(-(N+1) + 1)。这显然会导致neat trick。

【讨论】:

必须向下滚动到马特的评论才能看到它的正确使用;)【参考方案4】:

第一个 ~ 运算符将操作数强制为整数(可能在将值强制为字符串或布尔值之后),然后反转最低 31 位。官方 ECMAScript 数字都是浮点数,但在 SpiderMonkey 引擎中,有些数字是 31 位整数实现的。

您可以使用它将 1 元素数组转换为整数。浮点数根据 C 规则进行转换,即。小数部分的截断。

第二个 ~ 运算符然后将位反转,所以你知道你将有一个整数。这与在条件语句中将值强制为布尔值不同,因为空对象 的计算结果为 true,而 ~~ 的计算结果为 false。

js>~~"yes"
0
js>~~3
3
js>~~"yes"
0
js>~~false
0
js>~~""
0
js>~~true
1
js>~~"3"
3
js>~~
0
js>~~a:2
0
js>~~[2]
2
js>~~[2,3]
0
js>~~toString: function() return 4
4
js>~~NaN
0
js>~~[4.5]
4
js>~~5.6
5
js>~~-5.6
-5

【讨论】:

感谢 Shanti 提供的所有示例,这真的很有帮助! 还有~~undefined // 0 还有~~null // 0 从技术上讲,您的顺序有误。第二个~ 执行您描述的第一个~ 的操作,反之亦然。 ~ 运算符是一元运算符,从右到左进行交互 ~~X 就像 ~(~X) 不像 (~~)X (这将是一个语法错误)【参考方案5】:

在 ECMAScript 6 中,~~ 的等价物是 Math.trunc:

通过删除任何小数位返回数字的整数部分。它不会对任何数字进行四舍五入。

Math.trunc(13.37)   // 13
Math.trunc(42.84)   // 42
Math.trunc(0.123)   //  0
Math.trunc(-0.123)  // -0
Math.trunc("-1.123")// -1
Math.trunc(NaN)     // NaN
Math.trunc("foo")   // NaN
Math.trunc()        // NaN

polyfill:

function trunc(x) 
    return x < 0 ? Math.ceil(x) : Math.floor(x);

【讨论】:

有点令人惊讶的是,~~ 比 Math.trunc 快,jsperf.com/math-trunc-vs-double-bitwise-not-operator。不过,并非一切都与速度有关。可读性也很好。 ~~ 和 Math.trunc 之间有一个重要的区别:如果你传递一个字符串,或者 NaN 或者任何不是数字的东西,Math.trunc 将返回 NaN,而 ~~ 将始终返回一个数字,在这种情况下,它将返回 0。 根据jsperf.com/math-trunc-vs-double-bitwise-not-operator,Math.trunc 在 Chrome 59+ 中略快于 ~~。【参考方案6】:

这是一个如何有效使用此运算符的示例,在哪里使用它是有意义的:

leftOffset = -(~~$('html').css('padding-left').replace('px', '') + ~~$('body').css('margin-left').replace('px', '')),

来源:

See section Interacting with points

【讨论】:

【参考方案7】:

只是一点警告。这里的其他答案给我带来了一些麻烦。

目的是删除浮点数小数点后的任何内容,但它有一些极端情况,使其成为错误危险。我建议避免~~。

首先,~~ 不适用于非常大的数字。

~~1000000000000 == -727279968

作为替代方案,使用 Math.trunc()(正如 Gajus 所提到的,Math.trunc() 返回浮点数的整数部分,但仅在兼容 ECMAScript 6 的 JavaScript 中可用)。您始终可以通过以下方式为非 ECMAScript-6 环境创建自己的 Math.trunc()

if(!Math.trunc)
    Math.trunc = function(value)
        return Math.sign(value) * Math.floor(Math.abs(value));
    

我为此写了一篇博文供参考:http://bitlords.blogspot.com/2016/08/the-double-tilde-x-technique-in.html

【讨论】:

【参考方案8】:

波浪号(~) 有一个算法 -(N+1)

例如:

~0 = -(0+1) = -1
~5 = -(5+1) = -6
~-7 = -(-7+1) = 6

双波浪号是-(-(N+1)+1)

例如:

~~5 = -(-(5+1)+1) = 5
~~-3 = -(-(-3+1)+1) = -3

三个波浪号是 -(-(-(N+1)+1)+1)

例如:

~~~2 = -(-(-(2+1)+1)+1) = -3
~~~3 = -(-(-(3+1)+1)+1) = -4

【讨论】:

【参考方案9】:

将字符串转换为数字

console.log(~~-1);    // -1
console.log(~~0);     // 0
console.log(~~1);     // 1
console.log(~~"-1");  // -1
console.log(~~"0");   // 0
console.log(~~"1");   // 1
console.log(~~true);  // 1
console.log(~~false); // 0

~-1 为 0

if (~someStr.indexOf("a")) 
  // Found it
 else  
  // Not Found

source

【讨论】:

【参考方案10】:

~~ 可以用作Math.trunc() 的简写

~~8.29 // output 8

Math.trunc(8.29) // output 8

【讨论】:

以上是关于~~(“双波浪号”)在 Javascript 中做了啥?的主要内容,如果未能解决你的问题,请参考以下文章

<!-- 在Javascript中做啥?

[].forEach.call() 在 JavaScript 中做了啥?

“|”是啥意思(单管道)在 JavaScript 中做啥?

如何在 JavaScript 对象中做 css

“use strict”在 JavaScript 中做了啥,背后的原因是啥?

“var”在 JavaScript 中做了啥?为啥有时它是作业的一部分? [复制]