在Web
开发过程中,有时候我们需要将一些用二进制流表示的图片,转化为Base64
格式,让图片可以在img
标签中显示,下面将展示如何使用js
编码将图片转换为Base64
格式。
1. Binary To Base64
其实只需要一行代码就可以实现
// arrayBuffer为接收到的二进制数据
let imgBase64 = 'data:image/png;base64,' + window.btoa(String.fromCharCode(...new Uint8Array(arrayBuffer)));
但是,如果图片比较大(图片大于一百多k的时候),使用这种方法进行转换时,会抛出错误Uncaught RangeError: Maximum call stack size exceeded
,这是因为String.fromCharCode
方法无法一次处理这么大的数据,可以替换为下面的代码
// blob为接收到的二进制数据
let imgBase64 = window.btoa(new Uint8Array(arrayBuffer).reduce(function (data, byte) {
return data + String.fromCharCode(byte);
}, ''));
使用reduce来依次进行拼接
2. Data URLs
在上面中解决方法中,类似data:image/png;base64,
这样的前缀,作为开发人员,我们经常会看到,但是我们可能从来都没有考虑过它的含义。
实际上,
Data URLs,即前缀为 data:
协议的URL,其允许内容创建者向文档中嵌入小文件。
这里的data:
和http
、https:
等类似,格式如下
data:[<mediatype>][;base64],<data>
其中mediatype
与;base64
都是可选的,这里不要忘记,
,否则无法解析。
常见的示例如下所示
data:,文本数据
data:text/plain,文本数据
data:text/css,CSS代码
data:text/css;base64,base64编码的CSS代码
data:text/javascript,Javascript代码
data:text/javascript;base64,base64编码的Javascript代码
编码的gif图片数据
编码的png图片数据
编码的jpeg图片数据
编码的icon图片数据
上面的示例,在编码正确的情况下,都可以输入浏览器地址栏获取结果,注意中文会出现乱码
3. btoa
& atob
IE10+
的浏览器开始,所有浏览器都原生提供了base64
的解码和编码方法,即window.atob()
和window.btoa()
。
与Base64
库的功能类似,但是原生的base64
解码和编码方法在旧版本的浏览器中存在兼容性问题,所以为了解决兼容性问题,在旧版本的浏览器中,依旧需要使用Base64
库,或者自己实现。
4. ArrayBuffer
首先看一个使用场景
//sourceUrl 可以是本地服务器文件,也可以是远程服务器文件
//eg
//服务器根目录有mm.jpg文件,那么sourceUrl=mm.jpg,实际请求时请求的路径为http://localhost:port/mm.jpg
//fetch
const response = await fetch(sourceUrl, {method: 'GET'})
const responseData = await response.arrayBuffer()
let imgBase64 = window.btoa(new Uint8Array(responseData).reduce(function (data, byte) {
return data + String.fromCharCode(byte);
}, ''));
ArrayBuffer
是无法直接被使用的,需要通过TypedArray
或DataView
对象进行操作,它们会将缓冲区中的数据表示为特定的格式,进而通过这些特定的格式来读写缓冲区的内容。
这里之所以要用Uint8Array
作为TypedArray
而不是Uint16
等,是因为window.btoa
创建的是ASCII
字符串,而ASCII
的范围就是0~255
,所以用Uint8
5. 转换过程
- 获取
ArrayBuffer
对象 - 转换为
Uint8Array
对象 - 转换为普通字符串
- 转换为
base64
编码的字符串 - 拼接为
Data URL
格式
参考资料
