PCYOPCYO

山上有座庙

神奇的两次按位非运算符

谈到了连续进行两次按位非运算相当于floor的结果.

然后我进行了下测试,下面是我的代码:


 这个在我机器的运行的结果是1234567891011$time=microtime(1);for($i=0;$i<=100000;$i++){ ~~4.9;}echomicrotime(1)-$time;echo"<br>";$time=microtime(1);for($i=0;$i<=100000;$i++){floor(4.9);}echomicrotime(1)-$time;

0.013797998428345
0.041538000106812

我们可以看到 用~~几乎比floor快了三倍..

但是我们需要知道为什么~~就等价于floor了.

这里需要涉及到一个进制转换并且取非的过程.

现在我们需要一个假定条件.我们的变量为有符号的整数5.

我们现在给它进行按位取非的运算.

5的二进制为 0000 0101

按位非之后为 1111 1010

这时候第一位符号位1表示负数. 负数需要本身取反并+1

其他位取反为 000 0101 +1之后的结果为 000 0110

加上前面的符号 所以就为-6了.

这时候我们我们逐渐的就会发现一个著名的公式 ~num = -(num + 1)

回到上面的话题.如果上面不是整数,而且是小数5.9呢?

那么Zend引擎在计算取非的时候就会把小数部分的二进制的舍弃掉.

所以~5.9就会得到-6了.

然后再进行一次按位非..按照上面的公式 我们就得到5了..这样的效果就和floor()是一样的.

什么.你再担心用~~而舍弃floor会出问题吗?

这个答案是确定的

比如你可以用下面的代码测试

123echo ~~99999999999999.99;echo"<br />";echofloor(99999999999999.99);

在我的计算机上运行的结果是

276447231
99999999999999

我们可以看到.用~~的时候很明显的溢出了..

所以在比较小的数字运算中..还是放心的使用~~来代替floor吧..

http://www.skiyo.cn/2010/06/07/double-bitwise-not/

本原创文章未经允许不得转载 | 当前页面:PCYO » 神奇的两次按位非运算符

评论