小编典典

在 JavaScript 中从 Base64 字符串创建 BLOB

all

我在字符串中有 Base64 编码的二进制数据:

const contentType = 'image/png';
const b64Data = 'iVBORw0KGgoAAAANSUhEUgAAAAUAAAAFCAYAAACNbyblAAAAHElEQVQI12P4//8/w38GIAXDIBKE0DHxgljNBAAO9TXL0Y4OHwAAAABJRU5ErkJggg==';

我想创建一个blob:包含此数据的 URL 并将其显示给用户:

const blob = new Blob(????, {type: contentType});
const blobUrl = URL.createObjectURL(blob);

window.location = blobUrl;

我一直无法弄清楚如何创建 BLOB。

在某些情况下,我可以通过使用data:URL 来避免这种情况:

const dataUrl = `data:${contentType};base64,${b64Data}`;

window.location = dataUrl;

但是,在大多数情况下,data:URL 都非常大。


如何在 JavaScript 中将 Base64 字符串解码为 BLOB 对象?


阅读 139

收藏
2022-03-07

共1个答案

小编典典

atob函数会将 Base64 编码的字符串解码为一个新字符串,其中二进制数据的每个字节都有一个字符。

const byteCharacters = atob(b64Data);

每个字符的代码点 (charCode) 将是字节的值。.charCodeAt我们可以通过使用字符串中每个字符的方法应用它来创建一个字节值数组。

const byteNumbers = new Array(byteCharacters.length);
for (let i = 0; i < byteCharacters.length; i++) {
    byteNumbers[i] = byteCharacters.charCodeAt(i);
}

您可以通过将这个字节值数组传递给Uint8Array构造函数来将其转换为真正的类型化字节数组。

const byteArray = new Uint8Array(byteNumbers);

反过来,可以通过将其包装在数组中并将其传递给Blob构造函数来将其转换为 BLOB。

const blob = new Blob([byteArray], {type: contentType});

上面的代码有效。byteCharacters然而,通过处理更小的切片而不是一次处理,可以稍微提高性能。在我的粗略测试中,512
字节似乎是一个很好的切片大小。这为我们提供了以下功能。

const b64toBlob = (b64Data, contentType='', sliceSize=512) => {
  const byteCharacters = atob(b64Data);
  const byteArrays = [];

  for (let offset = 0; offset < byteCharacters.length; offset += sliceSize) {
    const slice = byteCharacters.slice(offset, offset + sliceSize);

    const byteNumbers = new Array(slice.length);
    for (let i = 0; i < slice.length; i++) {
      byteNumbers[i] = slice.charCodeAt(i);
    }

    const byteArray = new Uint8Array(byteNumbers);
    byteArrays.push(byteArray);
  }

  const blob = new Blob(byteArrays, {type: contentType});
  return blob;
}



const blob = b64toBlob(b64Data, contentType);
const blobUrl = URL.createObjectURL(blob);

window.location = blobUrl;

完整示例:

const b64toBlob = (b64Data, contentType='', sliceSize=512) => {

  const byteCharacters = atob(b64Data);

  const byteArrays = [];



  for (let offset = 0; offset < byteCharacters.length; offset += sliceSize) {

    const slice = byteCharacters.slice(offset, offset + sliceSize);



    const byteNumbers = new Array(slice.length);

    for (let i = 0; i < slice.length; i++) {

      byteNumbers[i] = slice.charCodeAt(i);

    }



    const byteArray = new Uint8Array(byteNumbers);

    byteArrays.push(byteArray);

  }



  const blob = new Blob(byteArrays, {type: contentType});

  return blob;

}



const contentType = 'image/png';

const b64Data = 'iVBORw0KGgoAAAANSUhEUgAAAAUAAAAFCAYAAACNbyblAAAAHElEQVQI12P4//8/w38GIAXDIBKE0DHxgljNBAAO9TXL0Y4OHwAAAABJRU5ErkJggg==';



const blob = b64toBlob(b64Data, contentType);

const blobUrl = URL.createObjectURL(blob);



const img = document.createElement('img');

img.src = blobUrl;

document.body.appendChild(img);
2022-03-07