我有一个以十六进制数表示的有符号值,例如0xffeb
,并希望将其转换为-21
作为“普通”Javascript整数。
到目前为止,我已经编写了一些代码:
function toBinary(a) { //: String
var r = '';
var binCounter = 0;
while (a > 0) {
r = a%2 + r;
a = Math.floor(a/2);
}
return r;
}
function twoscompl(a) { //: int
var l = toBinaryFill(a).length;
var msb = a >>> (l-1);
if (msb == 0) {
return a;
}
a = a-1;
var str = toBinary(a);
var nstr = '';
for (var i = 0; i < str.length; i++) {
nstr += str.charAt(i) == '1' ? '0' : '1';
}
return (-1)*parseInt(nstr);
}
问题是,我的函数为两个数字返回1作为MSB,因为只查看二进制表示“字符串”的MSB。对于这种情况,两个数字都是1:
-21 => 0xffeb => 1111 1111 1110 1011
21 => 0x15 => 1 0101
您是否有任何想法更有效,更好地实现这一点?
问候, mythbu
答案 0 :(得分:11)
使用parseInt()
进行转换(只接受十六进制字符串):
parseInt(a);
然后使用掩码判断MSB是否已设置:
a & 0x8000
如果返回非零值,则表示它为负数。
将它全部包装起来:
a = "0xffeb";
a = parseInt(a, 16);
if ((a & 0x8000) > 0) {
a = a - 0x10000;
}
请注意,这仅适用于16位整数(C中为short
)。如果你有一个32位整数,你需要一个不同的掩码和减法。
答案 1 :(得分:9)
我想出了这个
function hexToInt(hex) {
if (hex.length % 2 != 0) {
hex = "0" + hex;
}
var num = parseInt(hex, 16);
var maxVal = Math.pow(2, hex.length / 2 * 8);
if (num > maxVal / 2 - 1) {
num = num - maxVal
}
return num;
}
用法:
var res = hexToInt("FF"); // -1
res = hexToInt("A"); // same as "0A", 10
res = hexToInt("FFF"); // same as "0FFF", 4095
res = hexToInt("FFFF"); // -1
所以基本上十六进制转换范围取决于十六进制的长度,这就是我所寻找的。希望它有所帮助。
答案 2 :(得分:0)
function hexToSignedInt(hex) {
if (hex.length % 2 != 0) {
hex = "0" + hex;
}
var num = parseInt(hex, 16);
var maxVal = Math.pow(2, hex.length / 2 * 8);
if (num > maxVal / 2 - 1) {
num = num - maxVal
}
return num;
}
function hexToUnsignedInt(hex){
return parseInt(hex,16);
}
第一个用于有符号整数和 第二个是无符号整数
答案 3 :(得分:0)
基于@ Bart Friederichs,我附带:
function HexToSignedInt(num, numSize) {
var val = {
mask: 0x8 * Math.pow(16, numSize-1), // 0x8000 if numSize = 4
sub: -0x1 * Math.pow(16, numSize) //-0x10000 if numSize = 4
}
if(parseInt(num, 16) & val.mask > 0) { //negative
return (val.sub + parseInt(num, 16))
}else { //positive
return (parseInt(num,16))
}
}
所以现在您可以指定确切的长度(以半字节为单位)。
var numberToConvert = "CB8";
HexToSignedInt(numberToConvert, 3);
//expected output: -840
答案 4 :(得分:0)
因为我必须将绝对数值转换为范围从 -2^24 到 2^24-1 的 int32 值,
我想出了这个解决方案,您只需通过 parseInt(hex, 16)
将输入更改为数字,在您的情况下,nBytes 为 2。
function toSignedInt(value, nBytes) { // 0 <= value < 2^nbytes*4, nBytes >= 1,
var hexMask = '0x80' + '00'.repeat(nBytes - 1);
var intMask = parseInt(hexMask, 16);
if (value >= intMask) {
value = value - intMask * 2;
}
return value;
}
var vals = [ // expected output
'0x00', // 0
'0xFF', // 255
'0xFFFFFF', // 2^24 - 1 = 16777215
'0x7FFFFFFF', // 2^31 -1 = 2147483647
'0x80000000', // -2^31 = -2147483648
'0x80000001', // -2^31 + 1 = -2147483647
'0xFFFFFFFF', // -1
];
for (var hex of vals) {
var num = parseInt(hex, 16);
var result = toSignedInt(num, 4);
console.log(hex, num, result);
}
var sampleInput = '0xffeb';
var sampleResult = toSignedInt(parseInt(sampleInput, 16), 2);
console.log(sampleInput, sampleResult); // "0xffeb", -21