当前位置:   article > 正文

前端Ajax请求从后端获取二进制文件并下载

前端Ajax请求从后端获取二进制文件并下载

大家都知道前端的下载除了最简单的a标签href,还有时候需要验证token,此时后台会给一个返回二进制的下载接口。如果你用ajax普通的get,post请求,接口会返回乱码。那么本文就带你封装一个处理二进制下载的方法。

1.设置responseType为arraybuffer

这是正确获取二进制的关键,否则会被当成json文本来解析。

  1. const response = await axios({
  2. method,
  3. url,
  4. data,
  5. responseType: 'arraybuffer',
  6. headers,
  7. });

2.判断是否下载成功 

只有下载成功的时候返回的才是arraybuffer,否则是包含错误信息的json,因此在这里我们通过响应头的contentType来判断。

  1. if (contentType?.includes('application/json')) {
  2. // 响应的是json则提示错误信息
  3. const res = JSON.parse(new TextDecoder('utf-8').decode(new Uint8Array(response.data)));
  4. if (res.code !== 200) {
  5. ElMessage({
  6. message: res.msg,
  7. type: 'error',
  8. duration: 5 * 1000
  9. });
  10. return;
  11. }
  12. }

3.获取blobUrl 

  1. // blobType可以是空对象{},或指定的excel等MIME类型
  2. const data = new Blob([response.data], blobType);
  3. const src = window.URL.createObjectURL(data);

4.下载文件 

传入获取到的blobUrl,可以用第三方库file-saver下载,也可以用a标签的download属性。file-saver对各浏览器做了blob等兼容处理。

  1. import { saveAs } from 'file-saver';
  2. // 第一种 使用第三方库 file-saver
  3. saveAs(src, filename);
  4. // 第二种 a标签
  5. function aTagDownload(src, filename) {
  6. const link = document.createElement('a');
  7. link.href = src;
  8. link.setAttribute('download', filename);
  9. document.body.appendChild(link);
  10. link.click();
  11. document.body.removeChild(link);
  12. window.URL.revokeObjectURL(src);
  13. }

完整代码

  1. /**
  2. * 下载二进制文件
  3. * @param {string} method 必填 请求方式
  4. * @param {string} url 必填 下载 url
  5. * @param {object} [data={}] post 请求的 data
  6. * @param {object} [headers={}] 请求的 headers
  7. * @param {string} [filename=下载.zip] 保存的文件名,默认为下载.zip
  8. * @param {boolean} [isDownload=true] 是否直接下载,默认为 true
  9. * @param {object} [blobType={}] 指定 blob MIME 类型,默认为{}
  10. * @returns {string} blobUrl
  11. */
  12. export async function getBufferFile(
  13. method,
  14. url,
  15. data = {},
  16. headers = {},
  17. filename = '下载.zip',
  18. isDownload = true,
  19. blobType = {},
  20. ) {
  21. headers = {
  22. ...headers,
  23. Authorization: localStorage.getItem('token'),
  24. };
  25. ElMessage.success('已开始下载');
  26. try {
  27. const response = await axios({
  28. method,
  29. url,
  30. data,
  31. responseType: 'arraybuffer',
  32. headers,
  33. });
  34. const contentType = response.headers['content-type'];
  35. // 根据响应头的contentType判断是否下载成功
  36. if (contentType?.includes('application/json')) {
  37. // 响应的是json则提示错误信息
  38. const res = JSON.parse(new TextDecoder('utf-8').decode(new Uint8Array(response.data)));
  39. if (res.code !== 200) {
  40. ElMessage({
  41. message: res.msg,
  42. type: 'error',
  43. duration: 5 * 1000
  44. });
  45. return;
  46. }
  47. } else {
  48. const data = new Blob([response.data], blobType);
  49. const src = window.URL.createObjectURL(data);
  50. // 从响应头获取文件名
  51. if (response.headers['content-disposition']) {
  52. filename = decodeURI(
  53. response.headers['content-disposition'].split('filename=')[1]
  54. );
  55. }
  56. if (isDownload) {
  57. saveAs(src, filename); // 使用第三方库 file-saver
  58. }
  59. return src;
  60. }
  61. } catch (error) {
  62. console.error('下载文件失败:', error);
  63. ElMessage.error('下载文件失败');
  64. }
  65. }

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/凡人多烦事01/article/detail/334042?site
推荐阅读
相关标签
  

闽ICP备14008679号