Javascript: 位操作中的溢出和负数

本来遇到的问题是如何在Javascript中将十进制转变为十六进制(http://segmentfault.com/q/10100000000...)
第一次回答时我使用了这种方案:

var dec2hex = function(dec) {
  var buf = [],
  map = '0123456789ABCDEF';
  while (parseInt(dec / 16, 10) !== 0) {
    buf.unshift(map[dec % 16]);
    dec = parseInt(dec / 16, 10);
  }
  buf.unshift(map[dec % 16]);
  return buf.join('');
};

而在另一个我提出的问题中(http://segmentfault.com/q/10100000001...),我突然意识到还可以通过另一种方式进行转换:

var dec2hex2 = function(dec) {
  var hex = [],
  HEX = '0123456789ABCDEF';
  do {
    hex.unshift(HEX[dec & 0xF]);
  } while ( (dec = dec >> 4) !== 0 );
  return hex.join('');
};

知道最后google,我才发现原来事实比我想象的要简单得多。

var dec2hex3 = function(dec) {
  return dec.toString(16);
};

然而,问题来了。

第一次测试

输入:

var num = 542352345;
console.log(dec2hex(num));
console.log(dec2hex3(num));
console.log(dec2hex2(num));

输出:

2053A3D9
2053a3d9
2053A3D9

一切正常。

第二次测试

输入:

var num = 34542352345;
console.log(dec2hex(num));
console.log(dec2hex3(num));
console.log(dec2hex2(num));

输出:

80AE277D9
80ae277d9
AE277D9

Seemed weird?

第三次测试

输入:

var num = 5234542352345;
console.log(dec2hex(num));
console.log(dec2hex3(num));
console.log(dec2hex2(num));

输出:

4C2C30997D9
4c2c30997d9

这时候,dec2hex2函数似乎一直在运行,没有结果输出。

事先声明上述数字纯粹是我瞎打的。这时候我的第一个反应,是不是因为数字已经超过了32位(因为我的机器是32位的),然而我做了另外一个实验。
输入:

var num = parseInt(0xFFFFFFFF, 16);
console.log(dec2hex(num));
console.log(dec2hex3(num));
console.log(dec2hex2(num));

输出:

4294967295
4294967295

此时发生了和第三次测试同样的情况。

请问这是怎么回事呢?是不是发生了数字的溢出?那么如何避免和处理呢?

可能相关的资料(不确定)

https://dev.twitter.com/docs/twitter-...

阅读 9.4k
1 个回答

关键在于dec2hex2里用到的位移运算符。在ECMA规范中,位运算符总是针对32bit的整数进行的操作。至于死循环,是因为位移的时候会保留符号位。那个数字的bit32正好是1,所以一直右移最后得到的总是0xFFFFFFFF。

In ECMAScript (Javascript) bitwise operations are always in 32-bit. Therefore 5799218898 is chopped into 32-bit which becomes 1504251602. This integer >> 13 gives 183624.

Ref: http://stackoverflow.com/questions/23...

p.s. 建议,除非必要,避免使用位移运算符。你能想到的简单优化,编译器也很容易做到。

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题
宣传栏