如何将十六进制字符串转换为字节数组,以及十六进制字符串中的字节数组?

Posted

技术标签:

【中文标题】如何将十六进制字符串转换为字节数组,以及十六进制字符串中的字节数组?【英文标题】:How to convert hex string into a bytes array, and a bytes array in the hex string? 【发布时间】:2013-01-14 05:19:13 【问题描述】:

以下代码过程 bytes = parseHexString (createHexString (bytes)) 导致 更新字节,这是我想避免的。结果计算不正确。

<html>
<head>
<SCRIPT SRC="http://eu.static.mega.co.nz/sjcl_1.js"></SCRIPT>
<SCRIPT SRC="http://eu.static.mega.co.nz/crypto_1.js"></SCRIPT>
<SCRIPT SRC="http://eu.static.mega.co.nz/rsa_1.js"></SCRIPT>
<SCRIPT SRC="http://eu.static.mega.co.nz/hex_1.js"></SCRIPT>
<SCRIPT>
function parseHexString(str)  
    var result = [];
    while (str.length >= 2)  
        result.push(parseInt(str.substring(0, 2), 16));
        str = str.substring(2, str.length);
    

    return result;


function createHexString(arr) 
    var result = "";
    for (i in arr) 
        var str = arr[i].toString(16);
        str = str.length == 0 ? "00" :
              str.length == 1 ? "0" + str : 
              str.length == 2 ? str :
              str.substring(str.length-2, str.length);
        result += str;
    
    return result;


function t()

    var json_k     = 'aOrP5yLtNQT53WMQfufSlA';
    var json_csid  = 'CABD6JUMldvI_eqP0537xl9P8x7kgk2OjOq99Fy7kosphj6AFUtlbwRRDpg4EIifXRLO6FNpdD22WwtUlJ_1Mgye2Y87trEqLCbhahuEFJVQNMDtNbIem7xY2ER9uF-cdgBXZWuzp7XIBybSh7W8MSUlv_eGS6LcLGJ81Q49dSzVhcswHTJ_IJl04p3c0axR6ZIJ8dH5bJ_vXvgQsypUVVtdfMacKhB9cXdEtRZ6iWLKCKqscXdo6CNXlbIdzRhro0gxfmhfB_miysFAiSQrbtuYnIgYBU3i9p3jRlPD4ti3CUcnj0SomV61w1aEYNvo56HPMUZlVkVHA7BFzvHGHo0J';
    var json_privk = 'K7LDtk2M2QhjJx_v_Hqf0LKUBaZx76U_vBDjQty9HpFDy2MntF5HxxuyHQ9-1HmXeYzbL1pZnAxsZ7LRUbDnkR6qtJVaGdWuQhrytkuq0l5zBp-O--gZxoQPRGTsVgVRdAvpsRTkQI_q8fxADLCe0womFxtvvnD_FJgjaMsm7vkYchXkoq33WWyHijb3JMkymjl0_GtiSamT0qEL6sm_l5Z1lehqBGUEHfYAa0ub8IDx_yqy2R9Nh8Lwzmz4s24sShVxjaNsMBlSE-sEvTziosnNWK1Zl_XUYadlENkweuIoxYx_lt8XIV71TzjEFuVTd-pXhzVlqePmIu3SM3bO1Kzq_DnGfB62RmzlmbtHU4iyw4Hd1wQFRhTeSRrvMjsMPFKN-SIIQU7CRNaMuaDxZbNZcOKhMg_h9mApM0rRS3VZaGZzFTL9rSaDMYHw4pL3aOkSFPMY3w785Tss7Zqwuo9HFUWUVbnYAb97JkgCohlMotORrMMtual1dQ4sG1sIYXyWTckAGGL0ZAGurhtSKiyz1m8Lb39pXPacqFh_nCHqqb2_RdrKTj0PdGZESKkU8YedeqC1I9nR4v38DuQc-pBBR5DOwgNjJMvzvsUehs_PxIL8THjgIcr7ONc4hWV9o2v_l81Vo2cCW2I99Iz84IFN2fV1dTqHIG_tnLzz8ljBVygETUqrFdZ0JlQJkurZ7RBku5krm-k9CZmDezCIzPPil-RcYzVIk00gNYAxfiZE48Or4WEiGjgKLnHCYVtSlvlMF4bPGB4SVCZ-68j49EjfSWaMK0OoMkpGhqf7KchgxYBZq6o3AhLgp4t0BClvsdee6VTz1SFqc3m2A-TMG6fNdbCT_Q9nYCYdZIROdOc';

    var aes = new sjcl.cipher.aes( prepare_key_pw("oEyoo9cQcw") );
    k = decrypt_key(aes, base64_to_a32(json_k) );

    aes = new sjcl.cipher.aes(k);

    var t = mpi2b(base64urldecode(json_csid));

    var privk = a32_to_str(decrypt_key(aes,base64_to_a32(json_privk)));

    var rsa_privk = Array(4);
    for (var i = 0; i < 4; i++)
    
        var l = ((privk.charCodeAt(0)*256+privk.charCodeAt(1)+7)>>3)+2;

        rsa_privk[i] = mpi2b(privk.substr(0,l));
        if (typeof rsa_privk[i] == 'number') break;
        privk = privk.substr(l);    
    

    var p = parseHexString(createHexString(rsa_privk[0]));  // !!!!!!!!!!!!!!!!!!!!!!!!!!!!
    var q = parseHexString(createHexString(rsa_privk[1]));  // !!!!!!!!!!!!!!!!!!!!!!!!!!!!
    var d = parseHexString(createHexString(rsa_privk[2]));  // !!!!!!!!!!!!!!!!!!!!!!!!!!!!
    var u = parseHexString(createHexString(rsa_privk[3]));  // !!!!!!!!!!!!!!!!!!!!!!!!!!!!
    sid = base64urlencode(b2s(RSAdecrypt(t,d,p,q,u)).substr(0,43));
    if (sid!=='tajetAbW0qTQGFlwp8iD5lQ0TFV1QUZJZFVvjRX7Xx-bPzYBoau7qog09w')
        console.log("ERROR");

    p = rsa_privk[0];
    q = rsa_privk[1];
    d = rsa_privk[2];
    u = rsa_privk[3];
    sid = base64urlencode(b2s(RSAdecrypt(t,d,p,q,u)).substr(0,43));
    if (sid=='tajetAbW0qTQGFlwp8iD5lQ0TFV1QUZJZFVvjRX7Xx-bPzYBoau7qog09w')
        console.log("OK");

</script>

</head>
<body onload="t();"></body>
</html>

我不是 javascript 开发人员,并且在 google 代码中没有找到对这些数据起作用的

更新 1

console.log(createHexString(rsa_privk[0])); = e5d109c673d8ef03df564beb9e36e9983a23842b0a724efa45ff76bbe5ad72ed62d2757968 

但如果这样做

parseHexString('e5d109c673d8ef03df564beb9e36e9983a23842b0a724efa45ff76bbe5ad72ed‌​62d2757968'); 

然后代码if (sid!== ... 出错

更新 2

console.log(rsa_privk[0].toString(16));

输出: 123676133,198914513,129998601,245147334,11918451,206998232,96766191,75984899,177840095,106709334,10180427,208237547,119814814,127003446,189062377,84099480,220452154,250519075,267883908,115471915,165124106,238628722,169382478,42320122,95982405, 80725759,89608310,85166267,200925925,254033325,86971506,191278317,127411298,180195794,142776693,188738169,39016

更新 3

console.log(parseHexString(createHexString(rsa_privk[0])));
console.log(rsa_privk[0]);

输出:

[229, 209, 9, 198, 115, 216, 239, 3, 223, 86, 75, 235, 158, 54, 233, 152, 58, 35, 132, 43, 10, 114, 78, 250, 69, 255, 118, 187, 229, 173, 114, 237, 98, 210, 117, 121, 104]

[123676133, 198914513, 129998601, 245147334, 11918451, 206998232, 96766191, 75984899, 177840095, 106709334, 10180427, 208237547, 119814814, 127003446, 189062377, 84099480, 220452154, 250519075, 267883908, 115471915, 165124106, 238628722, 169382478, 42320122, 95982405, 80725759, 89608310, 85166267, 200925925, 254033325, 86971506, 191278317, 127411298, 180195794, 142776693, 188738169, 39016]

【问题讨论】:

您期望得到什么数据,得到什么数据? 我希望转换数据 = parseHexString(createHexString(data)) 不会改变数据 复制粘贴到createHexString(rsa_privk[0]),即console.log(rsa_privk[0]); console.log(createHexString(rsa_privk[0])); = e5d109c673d8ef03df564beb9e36e9983a23842b0a724efa45ff76bbe5ad72ed62d2757968 但是如果做 parseHexString('e5d109c673d8ef03df564beb9e36e9983a23842b0a724'efa4dff76b)2;276b;然后编码 if (sid!== ... make error 查看我的更新答案,只需添加console.log 行并将整个输出复制粘贴到您的问题中。 【参考方案1】:

将十六进制字符串转换为字节数组,反之亦然

注意:来自 crypto-js

的实现
// Convert a hex string to a byte array
function hexToBytes(hex) 
    for (var bytes = [], c = 0; c < hex.length; c += 2)
        bytes.push(parseInt(hex.substr(c, 2), 16));
    return bytes;


// Convert a byte array to a hex string
function bytesToHex(bytes) 
    for (var hex = [], i = 0; i < bytes.length; i++) 
        var current = bytes[i] < 0 ? bytes[i] + 256 : bytes[i];
        hex.push((current >>> 4).toString(16));
        hex.push((current & 0xF).toString(16));
    
    return hex.join("");

【讨论】:

这行得通。没有其他东西对我有用。非常感谢。 提醒一下,crypto-js api已经改成CryptoJS.enc.Hex.parse(),没有hexToBytes 代码与原来的略有不同。这是最新的 bytesToHex (stringify) 和 hexToBytes (parse) 方法的链接:crypto-js/core.js#L390 bytesToHex 在某些情况下计算很奇怪。例如[159,15] 被计算为 '9f0f' 它应该是 '9ff'。 [25,255] 确实再次起作用。这是字节序的事情吗?问题从第一个数组元素的第 9 位开始。接受的答案中的函数 createHexString 对我有用。 @TefoD 我无法重现您的问题。该代码对我有用【参考方案2】:

更新:向下滚动以获取解决方案...Live Demo

问题:您正在使用有损转换为十六进制,这是无法逆转的。

var p = parseHexString(createHexString(rsa_privk[0]));

这永远不会与rsa_privk[0] 相同。

因为,createHexString() 只使用每个数组元素中的last 2 bytes

例子:

rsa_privk[0] : [123676133, 198914513, 129998601, 245147334, 11918451, 206998232, 96766191, 75984899, 177840095, 106709334, 10180427, 208237547, 119814814, 127003446, 189062377, 84099480, 220452154, 250519075, 267883908, 115471915, 165124106, 238628722, 169382478, 42320122, 95982405, 80725759, 89608310, 85166267, 200925925, 254033325, 86971506, 191278317, 127411298, 180195794, 142776693, 188738169, 39016]

createHexString(rsa_privk[0]) : e5d109c673d8ef03df564beb9e36e9983a23842b0a724efa45ff76bbe5ad72ed62d2757968

parseHexString(createHexString(rsa_privk[0])) : [229, 209, 9, 198, 115, 216, 239, 3, 223, 86, 75, 235, 158, 54, 233, 152, 58, 35, 132, 43, 10, 114, 78, 250, 69, 255, 118, 187, 229, 173, 114, 237, 98, 210, 117, 121, 104] 

更新:工作解决方案...

这两个函数...十六进制始终包含 8 个字节块,每个块用于数组中的每个元素...

function parseHexString(str)  
    var result = [];
    while (str.length >= 8)  
        result.push(parseInt(str.substring(0, 8), 16));

        str = str.substring(8, str.length);
    

    return result;


function createHexString(arr) 
    var result = "";
    var z;

    for (var i = 0; i < arr.length; i++) 
        var str = arr[i].toString(16);

        z = 8 - str.length + 1;
        str = Array(z).join("0") + str;

        result += str;
    

    return result;

测试代码...

function test()    
    a = [123676133, 198914513, 129998601, 245147334, 11918451, 206998232, 96766191, 75984899, 177840095, 106709334, 10180427, 208237547, 119814814, 127003446, 189062377, 84099480, 220452154, 250519075, 267883908, 115471915, 165124106, 238628722, 169382478, 42320122, 95982405, 80725759, 89608310, 85166267, 200925925, 254033325, 86971506, 191278317, 127411298, 180195794, 142776693, 188738169, 39016];

    console.log("Input");
    console.log(a);

    b = createHexString(a);

    console.log("Hex");
    console.log(b);

    c = parseHexString(b); 

    console.log("Output");
    console.log(c);

    if(checkIfEqual(a, c)) 
        alert("Same");
    


function checkIfEqual(arr1, arr2) 
    if (arr1.length != arr2.length) 
        return false;
    
    //sort them first, then join them and just compare the strings
    return arr1.sort().join() == arr2.sort().join();

【讨论】:

你的输入输出是什么? 上述代码中的数据是二进制形式的。问题是如何以十六进制格式显示它们。但我肯定不会得到正确的结果,因为逆变换数据不正确。 你的数据是什么样的?在此处复制粘贴数据。 数据是二进制的,这里不能插入二进制数据!!!它们是在上面的代码中计算的。而且我不知道如何正确地将它们转换为十六进制并返回二进制 arr[i] 返回一个字符。你想要的是字符的字节数,你可以用 .charCodeAt( ... )【参考方案3】:

我只是想补充一下,https://github.com/dcodeIO/bytebuffer.js 有一个库可以轻松帮助进行这样的转换,因此您不需要编写自己的函数(这可能不是最佳的,或者是如果您的解决方案通过 GitHub 上的开源社区进行了审核,则更佳)。

var ByteBuffer = require("bytebuffer");

var bb = ByteBuffer.fromHex(yourHexString);

// need to convert it to base 64?
// bb.toBase64();

请参阅https://github.com/dcodeIO/bytebuffer.js/wiki/API#bytebufferfromhexstr-littleendian-noassert 以获取 API 文档以及对我上面使用的方法的更多见解。

【讨论】:

【参考方案4】:

澄清一下,如果您只是想对一个简单的字符串进行十六进制解码,例如48656C6C6F20576F726C6421 (Hello World!),您可以使用 OP 函数,但不要使用 8 的长度,而应该使用 2 的长度。

代码:

var DecodeHexStringToByteArray = function (hexString) 
   var result = [];
   while (hexString.length >= 2)  
       result.push(parseInt(hexString.substring(0, 2), 16));
       hexString = hexString.substring(2, hexString.length);
   
   return result;

输出将是[72, 101, 108, 108, 111, 32, 87, 111, 114, 108, 100, 33]

我知道这段代码已经在 OP 问题中,但它不在接受的答案中。我的目的只是直接回答所提问题的第一部分 (How to convert a hex string into a bytes array)。

【讨论】:

【参考方案5】:

我在这里找到了解决方案enter link description here

function hexStringToByteArray(hexString) 
    if (hexString.length % 2 !== 0) 
        throw "Must have an even number of hex digits to convert to bytes";
    
    var numBytes = hexString.length / 2;
    var byteArray = new Uint8Array(numBytes);
    for (var i=0; i<numBytes; i++) 
        byteArray[i] = parseInt(hexString.substr(i*2, 2), 16);
    
    return byteArray;

再次感谢http://www.java2s.com

【讨论】:

【参考方案6】:

这是此测试的实时样本。

http://jsfiddle.net/vincentwang2020/eks1z4g2/

function testcreateHexString()

    alert('test function createHexString');
    var key = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24];
    var result = createHexString(key);
    alert ('Hex value:' + result);

    alert('test function parseHexString');

    var key2 = parseHexString(result);

    if (key.sort().join() == key2.sort().join())
        alert ('Matched');


【讨论】:

以上是关于如何将十六进制字符串转换为字节数组,以及十六进制字符串中的字节数组?的主要内容,如果未能解决你的问题,请参考以下文章

如何将十六进制字符串转换为字节数组? [复制]

如何将字符串转换为十六进制字节数组? [复制]

如何将普通字符串转换为等效的十六进制字节数组? [复制]

如何在 C 中将字节数组转换为十六进制字符串?

在 Java 中,如何将字节数组转换为十六进制数字字符串,同时保持前导零? [复制]

在 Java 中,如何将字节数组转换为十六进制数字字符串,同时保持前导零? [复制]