在javascript中将字节数组转换为字符串

时间:2010-07-07 14:46:22

标签: javascript casting

如何将字节数组转换为字符串?

我发现这些功能正好相反:

function string2Bin(s) {
    var b = new Array();
    var last = s.length;

    for (var i = 0; i < last; i++) {
        var d = s.charCodeAt(i);
        if (d < 128)
            b[i] = dec2Bin(d);
        else {
            var c = s.charAt(i);
            alert(c + ' is NOT an ASCII character');
            b[i] = -1;
        }
    }
    return b;
}

function dec2Bin(d) {
    var b = '';

    for (var i = 0; i < 8; i++) {
        b = (d%2) + b;
        d = Math.floor(d/2);
    }

    return b;
}

但是如何让功能以其他方式运行?

感谢。

14 个答案:

答案 0 :(得分:69)

您需要将每个八位字节解析回数字,并使用该值来获取字符,如下所示:

function bin2String(array) {
  var result = "";
  for (var i = 0; i < array.length; i++) {
    result += String.fromCharCode(parseInt(array[i], 2));
  }
  return result;
}

bin2String(["01100110", "01101111", "01101111"]); // "foo"

// Using your string2Bin function to test:
bin2String(string2Bin("hello world")) === "hello world";

修改:是的,您当前的string2Bin可以写得更快:

function string2Bin(str) {
  var result = [];
  for (var i = 0; i < str.length; i++) {
    result.push(str.charCodeAt(i).toString(2));
  }
  return result;
}

但是通过查看你链接的文档,我认为setBytesParameter方法期望blob数组包含十进制数,而不是位字符串,所以你可以写一些像这样:

function string2Bin(str) {
  var result = [];
  for (var i = 0; i < str.length; i++) {
    result.push(str.charCodeAt(i));
  }
  return result;
}

function bin2String(array) {
  return String.fromCharCode.apply(String, array);
}

string2Bin('foo'); // [102, 111, 111]
bin2String(string2Bin('foo')) === 'foo'; // true

答案 1 :(得分:33)

简单apply您的字节数组为String.fromCharCode。例如

String.fromCharCode.apply(null, [102, 111, 111])  等于'foo'。

警告:适用于小于65535的数组。 MDN文档here

答案 2 :(得分:7)

即使 more 也可以简单地编写string2Bin,并且无需任何循环,即可启动!

function string2Bin ( str ) {
    return str.split("").map( function( val ) { 
        return val.charCodeAt( 0 ); 
    } );
}

答案 3 :(得分:6)

尝试使用新的文本编码API:

// create an array view of some valid bytes
let bytesView = new Uint8Array([104, 101, 108, 108, 111]);

console.log(bytesView);

// convert bytes to string
// encoding can be specfied, defaults to utf-8 which is ascii.
let str = new TextDecoder().decode(bytesView); 

console.log(str);

// convert string to bytes
// encoding can be specfied, defaults to utf-8 which is ascii.
let bytes2 = new TextEncoder().encode(str);

// look, they're the same!
console.log(bytes2);
console.log(bytesView);

答案 4 :(得分:3)

我认为这会更有效率:

function toBinString (arr) {
    var uarr = new Uint8Array(arr.map(function(x){return parseInt(x,2)}));
    var strings = [], chunksize = 0xffff;
    // There is a maximum stack size. We cannot call String.fromCharCode with as many arguments as we want
    for (var i=0; i*chunksize < uarr.length; i++){
        strings.push(String.fromCharCode.apply(null, uarr.subarray(i*chunksize, (i+1)*chunksize)));
    }
    return strings.join('');
}

答案 5 :(得分:1)

即使我有点迟了,我认为未来的用户分享我使用ES6做的一些单行实现会很有趣。

根据您的环境或/和您将对数据执行的操作,我认为重要的一件事是保留完整的字节值。例如,(5).toString(2)会为您提供101,但完整的二进制转换实际上是00000101,这就是为什么您可能需要创建leftPad实现来填充字符串的原因带前导零的字节。但你可能根本不需要它,就像其他答案一样。

如果你运行下面的代码片段,你会看到第一个输出是将abc字符串转换为字节数组,然后将所述数组重新转换为相应的字符串。< / p>

// For each byte in our array, retrieve the char code value of the binary value
const binArrayToString = array => array.map(byte => String.fromCharCode(parseInt(byte, 2))).join('')

// Basic left pad implementation to ensure string is on 8 bits
const leftPad = str => str.length < 8 ? (Array(8).join('0') + str).slice(-8) : str

// For each char of the string, get the int code and convert it to binary. Ensure 8 bits.
const stringToBinArray = str => str.split('').map(c => leftPad(c.charCodeAt().toString(2)))

const array = stringToBinArray('abc')

console.log(array)
console.log(binArrayToString(array))

答案 6 :(得分:1)

字符串数组: "FooBar".split('').map(c => c.charCodeAt(0));

字节数组到字符串: [102, 111, 111, 98, 97, 114].map(c => String.fromCharCode(c)).join('');

答案 7 :(得分:1)

这应该有效:

String.fromCharCode(...array);

String.fromCodePoint(...array)

答案 8 :(得分:1)

如果您使用的是 node.js ,则可以执行以下操作:

yourByteArray.toString('base64');

答案 9 :(得分:0)

找不到任何适用于UTF-8字符的解决方案。在遇到2个字节字符之前,String.fromCharCode很好。

例如Hüser将以[0x44,0x61,0x6e,0x69,0x65,0x6c,0x61,0x20,0x48,0xc3,0xbc,0x73,0x65,0x72]

的形式出现

但是如果您使用String.fromCharCode进行操作,则会有Hüser,因为每个字节将分别转换为char。

解决方案

目前,我正在使用以下解决方案:

function pad(n) { return (n.length < 2 ? '0' + n : n); }
function decodeUtf8(data) {
  return decodeURIComponent(
    data.map(byte => ('%' + pad(byte.toString(16)))).join('')
  );
}

答案 10 :(得分:0)

我有一些解密的字节数组,其中包含填充字符和其他不需要的东西,所以我做到了这一点(可能并不完美,但只能用于有限的用途)

var junk = String.fromCharCode.apply(null, res).split('').map(char => char.charCodeAt(0) <= 127 && char.charCodeAt(0) >= 32 ? char : '').join('');

答案 11 :(得分:0)

来不及回答,但是如果您输入的是ASCII字节形式,则可以尝试以下解决方案:

function convertArrToString(rArr){
 //Step 1: Convert each element to character
 let tmpArr = new Array();
 rArr.forEach(function(element,index){
    tmpArr.push(String.fromCharCode(element));
});
//Step 2: Return the string by joining the elements
return(tmpArr.join(""));
}

function convertArrToHexNumber(rArr){
  return(parseInt(convertArrToString(rArr),16));
}

答案 12 :(得分:0)

如果您的数组是使用UTF-8编码的,则您不能使用 TextDecoder API ,因为it is not supported on IE

  1. 您可以使用FastestSmallestTextEncoderDecoder建议的Mozilla Developer Network website polyfill;
  2. 您可以使用MDN website上提供的此功能:

function utf8ArrayToString(aBytes) {
    var sView = "";
    
    for (var nPart, nLen = aBytes.length, nIdx = 0; nIdx < nLen; nIdx++) {
        nPart = aBytes[nIdx];
        
        sView += String.fromCharCode(
            nPart > 251 && nPart < 254 && nIdx + 5 < nLen ? /* six bytes */
                /* (nPart - 252 << 30) may be not so safe in ECMAScript! So...: */
                (nPart - 252) * 1073741824 + (aBytes[++nIdx] - 128 << 24) + (aBytes[++nIdx] - 128 << 18) + (aBytes[++nIdx] - 128 << 12) + (aBytes[++nIdx] - 128 << 6) + aBytes[++nIdx] - 128
            : nPart > 247 && nPart < 252 && nIdx + 4 < nLen ? /* five bytes */
                (nPart - 248 << 24) + (aBytes[++nIdx] - 128 << 18) + (aBytes[++nIdx] - 128 << 12) + (aBytes[++nIdx] - 128 << 6) + aBytes[++nIdx] - 128
            : nPart > 239 && nPart < 248 && nIdx + 3 < nLen ? /* four bytes */
                (nPart - 240 << 18) + (aBytes[++nIdx] - 128 << 12) + (aBytes[++nIdx] - 128 << 6) + aBytes[++nIdx] - 128
            : nPart > 223 && nPart < 240 && nIdx + 2 < nLen ? /* three bytes */
                (nPart - 224 << 12) + (aBytes[++nIdx] - 128 << 6) + aBytes[++nIdx] - 128
            : nPart > 191 && nPart < 224 && nIdx + 1 < nLen ? /* two bytes */
                (nPart - 192 << 6) + aBytes[++nIdx] - 128
            : /* nPart < 127 ? */ /* one byte */
                nPart
        );
    }
    
    return sView;
}

let str = utf8ArrayToString([50,72,226,130,130,32,43,32,79,226,130,130,32,226,135,140,32,50,72,226,130,130,79]);

// Must show 2H₂ + O₂ ⇌ 2H₂O
console.log(str);

答案 13 :(得分:0)

我找到的最简单的解决方案是:

var text = atob(byteArray);
相关问题