PCYO 评测夜鸥 六度计算世界

少年强 则 中国强

神奇的两次按位非运算符

谈到了连续进行两次按位非运算相当于floor的结果.
然后我进行了下测试,下面是我的代码:

这个在我机器的运行的结果是1234567891011$time=[microtime](http://www.php.net/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](http://www.php.net/floor)(4.9);}echo[microtime](http://www.php.net/microtime)(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"
";echofloor(99999999999999.99);
在我的计算机上运行的结果是
276447231
99999999999999
我们可以看到.用~~的时候很明显的溢出了..
所以在比较小的数字运算中..还是放心的使用~~来代替floor吧..
http://www.skiyo.cn/2010/06/07/double-bitwise-not/

本原创文章未经允许不得转载 | 当前页面:PCYO 评测夜鸥 六度计算世界 » 神奇的两次按位非运算符

评论