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

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

1.设置responseType为arraybuffer

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

const response = await axios({
  method,
  url,
  data,
  responseType: 'arraybuffer',
  headers,
});

2.判断是否下载成功 

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

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

3.获取blobUrl 

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

4.下载文件 

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

import { saveAs } from 'file-saver';

// 第一种 使用第三方库 file-saver
saveAs(src, filename); 

// 第二种 a标签
function aTagDownload(src, filename) {
  const link = document.createElement('a');
  link.href = src;
  link.setAttribute('download', filename);
  document.body.appendChild(link);
  link.click();

  document.body.removeChild(link);
  window.URL.revokeObjectURL(src);
}

完整代码

/**
 * 下载二进制文件
 * @param {string} method 必填 请求方式
 * @param {string} url 必填 下载 url
 * @param {object} [data={}] post 请求的 data
 * @param {object} [headers={}] 请求的 headers
 * @param {string} [filename=下载.zip] 保存的文件名,默认为下载.zip
 * @param {boolean} [isDownload=true] 是否直接下载,默认为 true
 * @param {object} [blobType={}] 指定 blob MIME 类型,默认为{}
 * @returns {string} blobUrl
 */
export async function getBufferFile(
  method,
  url,
  data = {},
  headers = {},
  filename = '下载.zip',
  isDownload = true,
  blobType = {},
) {
  headers = {
    ...headers,
    Authorization: localStorage.getItem('token'),
  };

  ElMessage.success('已开始下载');

  try {
    const response = await axios({
      method,
      url,
      data,
      responseType: 'arraybuffer',
      headers,
    });
    const contentType = response.headers['content-type'];

    // 根据响应头的contentType判断是否下载成功
    if (contentType?.includes('application/json')) {
      // 响应的是json则提示错误信息
      const res = JSON.parse(new TextDecoder('utf-8').decode(new Uint8Array(response.data)));
      if (res.code !== 200) {
        ElMessage({
          message: res.msg,
          type: 'error',
          duration: 5 * 1000
        });
        return;
      }
    } else {
      const data = new Blob([response.data], blobType);
      const src = window.URL.createObjectURL(data);
      // 从响应头获取文件名
      if (response.headers['content-disposition']) {
        filename = decodeURI(
          response.headers['content-disposition'].split('filename=')[1]
        );
      }

      if (isDownload) {
        saveAs(src, filename); // 使用第三方库 file-saver
      }
      
      return src;
    }
  } catch (error) {
    console.error('下载文件失败:', error);
    ElMessage.error('下载文件失败');
  }
}

  • 15
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
首先,你需要在前端使用ajax后端发送请求,并在请求中设置responseType为blob,以便获取二进制流数据。同时,你需要在后端将excel文件转换成二进制流数据并返回给前端。 以下是一个使用jQuery的前端示例代码: ```javascript $.ajax({ url: 'your-backend-url', method: 'POST', responseType: 'blob', // 设置返回数据类型为二进制流 success: function(data) { // 创建一个a标签,将二进制流数据转换为URL,并设置下载属性 var a = document.createElement('a'); var url = window.URL.createObjectURL(data); a.href = url; a.download = 'file.xls'; // 将a标签添加到文档中,并自动触发点击事件下载文件 document.body.appendChild(a); a.click(); // 下载完成后,释放URL对象 window.URL.revokeObjectURL(url); } }); ``` 在后端,你可以使用EPPlus或NPOI等库来将Excel文件转换为二进制流数据,并将其作为响应返回给前端。以下是一个使用EPPlus的后端示例代码: ```csharp using OfficeOpenXml; // ... [HttpPost] public IActionResult DownloadExcel() { // 读取Excel文件 var fileInfo = new FileInfo("path/to/file.xlsx"); using (var package = new ExcelPackage(fileInfo)) { // 获取工作表 var worksheet = package.Workbook.Worksheets[0]; // 将工作表转换为二进制数据 var data = worksheet.ConvertToBinary(); // 返回二进制数据 return File(data, "application/vnd.ms-excel", "file.xls"); } } ``` 注意,以上代码仅供参考,具体实现可能会因为框架和库的不同而有所不同。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值