javascript中负数算术右移、逻辑右移的奥秘探索 |
本文标签:负数算术右移,逻辑右移 javascript中负数的算术右移和逻辑右移都十分的让人迷惑,特别是逻辑右移>>>,你会发现即使一个很小的负数,右移之后,也会得到一个无比巨大的数,这是为什么呢? 原来在逻辑右移中符号位会随着整体一起往右移动,这样就是相当于无符号数的移动了,最后得到的就是一个正数,因为符号位不存在了 。首先逻辑右移产生的一定是32位的数,然后负数的符号位为1,这意味着从第32位到符号位的位置全部由1填充,这样的数能不大吗例如-1,逻辑右移0位表现形式就是1111 1111 1111 1111 1111 1111 1111 1111 ,这样的数是当作正数来对待的!所以将-1逻辑右移N位,最后的结果都是全为1! 左移运算保留数字的符号位 。例如,如果把 -2 左移 5 位,得到的是 -64,而不是 64 。“符号仍然存储在第 32 位中吗?”是的,不过这在 ECMAScript 后台进行,开发者不能直接访问第 32 个数位 。即使输出二进制字符串形式的负数,显示的也是负号形式(例如,-2 将显示 -10 。) 有符号右移运算 有符号右移运算符由两个大于号表示(<$lt;) 。它把 32 位数字中的所有数位整体右移,同时保留该数的符号(正号或负号) 。有符号右移运算符恰好与左移运算相反 。例如,把 64 右移 5 位,将变为 2: var iOld = 64; //等于二进制 1000000 var iNew = iOld >> 5; //等于二进制 10 十进制 2同样,移动数位后会造成空位 。这次,空位位于数字的左侧,但位于符号位之后 。ECMAScript 用符号位的值填充这些空位,创建完整的数字,如下图所示: 无符号右移运算 无符号右移运算符由三个大于号(>>>)表示,它将无符号 32 位数的所有数位整体右移 。对于正数,无符号右移运算的结果与有符号右移运算一样 。 用有符号右移运算中的例子,把 64 右移 5 位,将变为 2: 无符号右移运算用 0 填充所有空位 。对于正数,这与有符号右移运算的操作一样,而负数则被作为正数来处理 。 由于无符号右移运算的结果是一个 32 位的正数,所以负数的无符号右移运算得到的总是一个非常大的数字 。例如,如果把 -64 右移 5 位,将得到 134217726 。如果得到这种结果的呢? 要实现这一点,需要把这个数字转换成无符号的等价形式(尽管该数字本身还是有符号的),可以通过以下代码获得这种形式: var iUnsigned64 = -64 >>> 0; 然后,用 Number 类型的 toString() 获取它的真正的位表示,采用的基为 2:代码如下: alert(iUnsigned64.toString(2)); 这将生成 11111111111111111111111111000000,即有符号整数 -64 的二进制补码表示,不过它等于无符号整数 4294967232 。 出于这种原因,使用无符号右移运算符要小心 。 现在来说一下负数的算术右移>>: 我们发现-9>>2=-3,为什么是-3呢? 首先符号位是不变的,不参加右移,然后在9右移的过程中,最低位为1的话,那么右移之后最低位仍然为1!这是很奇怪的 。 |