javascript

时间:2015-12-16 10:49:02

标签: javascript arrays bitcoin data-conversion

我有一个[4,-101,122,-41,-30,23,-28,3,..]形式的字节数组,我希望以6d69f597b217fa333246c2c8的形式转换 我正在使用以下功能

function toHexString(bytes) {
  return bytes.map(function(byte) {
    return (byte & 0xFF).toString(16)
  }).join('')
}

它给了我一个相同形式的字符串,但我怀疑它不是一个有效的转换,因为十六进制字符串比预期的短一点。我认为翻译应该得到“0a10a6dc”。 请告诉我,如果我错了或这是一个正确的转换,但也许我没有使用正确的字节数组

字节数组4,-127,45,126,58,-104,41,-27,-43,27,-35,100,-50,-77,93,-16,96,105,-101,-63,48,-105,49,-67,110,111,26,84,67,-89,-7,-50,10,-12,56,47,-49,-42,-11,-8,-96,-117,-78,97,-105,9,-62,-44,-97,-73,113,96,23,112,-14,-62,103,-104,90,-14,117,78,31,-116,-7

相应的转化4812d7e3a9829e5d51bdd64ceb35df060699bc1309731bd6e6f1a5443a7f9ceaf4382fcfd6f5f8a08bb261979c2d49fb771601770f2c267985af2754e1f8cf9

9 个答案:

答案 0 :(得分:41)

您缺少十六进制转换中的填充。您将要使用

function toHexString(byteArray) {
  return Array.from(byteArray, function(byte) {
    return ('0' + (byte & 0xFF).toString(16)).slice(-2);
  }).join('')
}

使每个字节转换为正好两个十六进制数字。您的预期输出为04812d7e3a9829e5d51bdd64ceb35df060699bc1309731bd6e6f1a5443a7f9ce0af4382fcfd6f5f8a08bb2619709c2d49fb771601770f2c267985af2754e1f8cf9

答案 1 :(得分:5)

如果输入类型为map(),则使用Uint8Array无法工作:map()的结果也是Uint8Array,但不能function toHexString(byteArray) { var s = '0x'; byteArray.forEach(function(byte) { s += ('0' + (byte & 0xFF).toString(16)).slice(-2); }); return s; } 保持字符串转换的结果。

  @IBAction func ShareClicked(_ sender: UIButton) -> Void {
        print("Hello Edit Button")

        let indexPath = IndexPath(row: 0, section: 0)
        let cell = TableView.cellForRow(at: indexPath) as! ProfileTableViewCell

        let activityVC = UIActivityViewController(activityItems: [cell.Books!, "I found this book", cell.Title!,"in the LitSwap app for :",cell.Price! ], applicationActivities: nil)
        activityVC.popoverPresentationController?.sourceView = self.view

        self.present(activityVC, animated: true, completion: nil)

    }

答案 2 :(得分:5)

使用Array.reduce()更简洁,更高效(参见https://jsperf.com/byte-array-to-hex-string)替代方案:

function toHexString(byteArray) {
  return byteArray.reduce((output, elem) => 
    (output + ('0' + elem.toString(16)).slice(-2)),
    '');
}

(也没有“& 0xFF”,因为在我看来,如果传入包含大于255的值的数组,输出应该搞乱,这样用户可以更容易地看到他们的输入是错误的。)

答案 3 :(得分:3)

由于这是Google首次将“ js字节转换为十六进制”,因此我需要一些时间来了解Bergi的功能,因此我重写了该功能并添加了一些注释,使我更容易理解:

function byteToHex(byte) {
  // convert the possibly signed byte (-128 to 127) to an unsigned byte (0 to 255).
  // if you know, that you only deal with unsigned bytes (Uint8Array), you can omit this line
  const unsignedByte = byte & 0xff;

  // If the number can be represented with only 4 bits (0-15), 
  // the hexadecimal representation of this number is only one char (0-9, a-f). 
  if (unsignedByte < 16) {
    return '0' + unsignedByte.toString(16);
  } else {
    return unsignedByte.toString(16);
  }
}

// bytes is an typed array (Int8Array or Uint8Array)
function toHexString(bytes) {
  // Since the .map() method is not available for typed arrays, 
  // we will convert the typed array to an array using Array.from().
  return Array.from(bytes)
    .map(byte => byteToHex(byte))
    .join('');
}

OP忘记为只能用4位显示的数字添加前导0

答案 4 :(得分:1)

您需要使用适当数量的前导零填充十六进制转换。

答案 5 :(得分:0)

所有先前的解决方案都可以使用,但是它们都需要创建许多字符串,并对创建的字符串进行串联和切片。我想,既然有类型化数组,就必须有一种更好的方法来处理它。我最初是使用node进行此操作的,然后注释掉了使用Buffer的行并将其更改为TypedArrays,以便它也可以在浏览器中使用。

代码更多,但速度要快得多,至少在我整理的jsperf中是如此。接受的答案中的字符串操作版本执行37000次操作/秒,而下面的代码管理317000次操作/秒。创建字符串对象有很多隐藏的开销。

function toHexString (byteArray) {
  //const chars = new Buffer(byteArray.length * 2);
  const chars = new Uint8Array(byteArray.length * 2);
  const alpha = 'a'.charCodeAt(0) - 10;
  const digit = '0'.charCodeAt(0);

  let p = 0;
  for (let i = 0; i < byteArray.length; i++) {
      let nibble = byteArray[i] >>> 4;
      chars[p++] = nibble > 9 ? nibble + alpha : nibble + digit;
      nibble = byteArray[i] & 0xF;
      chars[p++] = nibble > 9 ? nibble + alpha : nibble + digit;    
  }

  //return chars.toString('utf8');
  return String.fromCharCode.apply(null, chars);
}

答案 6 :(得分:0)

要保持代码整洁,可以使用现有的库,例如array-buffer-to-hex。示例:

const arrayBufferToHex = require('array-buffer-to-hex')
const crypto = require('crypto')

const bytes = crypto.randomBytes(10)

console.log(arrayBufferToHex(bytes)) // => "557f694f76c628fd6acb"

答案 7 :(得分:0)

将字节数组转换为十六进制数组时,我们必须考虑如何对它们进行带符号的数字处理。如果是这样,我们必须先将它们转换为十进制数。 signed numbers to decimal conversion。然后,我们可以使用.toString(16)方法将其转换为十六进制。

const hexArr = byteArr.map((byte) => {
    if (byte < 0) {
      byte = -((byte ^ 0xff) + 1); //converting 2s complement to a decimal number
    }
    //add padding at the start to ensure it's always 2 characters long otherwise '01' will be '1'
    return byte.toString(16).padStart(2, '0'); 
});

答案 8 :(得分:0)

这是 ArrayBuffer 的跨浏览器解决方案:

    function buf2hex(buffer) {
        var u = new Uint8Array(buffer),
            a = new Array(u.length),
            i = u.length;
        while (i--) // map to hex
            a[i] = (u[i] < 16 ? '0' : '') + u[i].toString(16);
        u = null; // free memory
        return a.join('');
    };