我有一个字符串中的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对象?

推荐答案

atob函数将一个Base64编码的字符串解码成一个新字符串,每个二进制数据字节对应一个字符.

const byteCharacters = atob(b64Data);

每个字符的代码点(charCode)将是字节的值.我们可以为字符串中的每个字符使用.charCodeAt方法来创建字节值array.

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

通过将此字节值数组传递给Uint8Array构造函数,可以将其转换为实型字节array.

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);

Javascript相关问答推荐

使用JavaScript重命名对象数组中的键

创建私有JS出口

为什么JavaScript双边字符串文字插值不是二次的?

如何通过onClick为一组按钮分配功能;

当promise 在拒绝处理程序被锁定之前被拒绝时,为什么我们会得到未捕获的错误?

如何判断属于多个元素的属性是否具有多个值之一

无法在nextjs应用程序中通过id从mongoDB删除'

将2D数组转换为图形

Snowflake JavaScript存储过程返回成功,尽管预期失败

如何从网站www.example.com获取表与Cheerio谷歌应用程序脚本

如何在使用rhandsontable生成表时扩展数字输入验证?

当输入字段无效时,我的应用程序不会返回错误

Nextjs 13.4 Next-Auth 4.2登录(&Quot;凭据&,{});不工作

TinyMCE 6导致Data:Image对象通过提供的脚本过度上载

一个实体一刀VS每个实体多刀S

如何在TransformControls模式下只保留箭头进行翻译?

TypeORM QueryBuilder限制联接到一条记录

我们是否可以在reactjs中创建多个同名的路由

在点击链接后重定向至url之前暂停

将Windows XP转换为原始数据以在html前端中显示