前端下载二进制流文件
前后端对接时,后端常有返回二进制流文件的情况,前端需要下载或预览(pdf、html、图片文件等)。下面介绍几种实现前端获取二进制流并下载或预览的方式
1.直接预览二进制流文件
如果返回的二进制流是一个pdf文件或图片文件。那么可以通过打开新标签页的方式进行文件预览。例如请求如下接口https://via.placeholder.com/150/24f355
,返回一个图片文件,前端可以使用window.open("https://via.placeholder.com/150/24f355")
来打开新标签页实现预览。
当然这种情况是接口并没有做其他处理,如果接口需要验证token
请求头使用window.open
是无法设置请求头的。下面会介绍如何解决需要token
的情况
2.使用Blob对象实现下载
使用Blob对象实现下载主要分为以下几步:
- 设置接口请求的
responseType
值为'Blob'
- 创建Blob对象
const blob = new Blob([data])
- 创建一个不显示的a标签,点击下载按钮时实际跳转a标签的地址,完成下载
1
2
3
4
5
6
7const link = document.createElement('a') // 创建a标签
link.download = '文件名称.后缀名' // 设置下载的文件名称
link.style.display = 'none'
link.href = window.URL.createObjectURL(blob) // 创建Blob对象的URL赋值给a标签
link.target = 'blank'
document.body.append(link) // 将a标签添加到页面中
link.click() // 点击a标签实现下载 - 释放URL对象下面就是一个具体的例子
1
2window.URL.revokeObjectURL(link.href)
document.body.removeChild(link)1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>blob下载文件</title>
</head>
<body>
<button id="blobDown">下载</button>
<script src="https://unpkg.com/axios/dist/axios.min.js"></script>
<script type="text/javascript">
document.getElementById("blobDown").onclick = function () {
axios.get('https://img1.baidu.com/it/u=2921118138,3990044703&fm=26&fmt=auto', {
responseType: 'blob',
}).then(res => {
console.log(res.data);
const blob = new Blob([res.data])
const link = document.createElement('a')
link.download = '王冰冰.jpg'
link.style.display = 'none'
link.href = window.URL.createObjectURL(blob)
link.target = 'blank'
document.body.append(link)
link.click()
})
}
</script>
</body>
</html>3.对blob下载二进制文件流的方法进行抽离
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23/**
*
* @param {*} data blod文件流
* @param {*} fileName 下载后的文件名称
*/
function downloadBlob(data, fileName) {
const blob = new Blob([data])
if ('download' in document.createElement('a')) {
// 非IE下载
const elink = document.createElement('a')
elink.download = fileName
elink.style.display = 'none'
elink.href = URL.createObjectURL(blob)
elink.target = 'blank'
document.body.appendChild(elink)
elink.click()
URL.revokeObjectURL(elink.href) // 释放URL 对象
document.body.removeChild(elink)
} else {
// IE10+下载
navigator.msSaveBlob(blob, fileName)
}
}4. 解决window.open无法预览需要请求头的接口返回的二进制流
实际开发中,后端的接口都会需要用户验证,通常是放在请求头某个参数或具体的请求参数中。
如果在请求头中,那么window.open预览文件流则无法配置请求头,这时可以用blob解决更多Blob相关信息可以查看:1
2
3
4
5
6
7
8
9
10
11
12axios.get('https://img1.baidu.com/it/u=2921118138,3990044703&fm=26&fmt=auto', {
responseType: 'blob',
headers: {'Authorization': 'tokenInfo'} // 这里设置对应的请求头
}).then(res => {
console.log(res.data);
// 创建Blob对象的时候,设置指定的文件格式。如果是pdf则设置 'type: "application/pdf"'
const blob = new Blob([res.data], {type: 'image/jpeg'})
const link = document.createElement('a')
link.href = window.URL.createObjectURL(blob)
link.target = 'blank'
link.click()
})
Blob MDN
评论