前端下载文件的方式

IT技术2年前 (2022)发布 IT大王
0

概要

在前端下载文件是个很通用的需求,一般后端会提供下载的方式有两种:

  1. 直接返回文件的网络地址(一般用在静态文件上,比如图片以及各种音视频资源等)
  2. 返回文件流(一般用在动态文件上,比如根据前端选择,导出不同的统计结果 excel 等)

第一种方式比较简单,但是使用场景有限。
第二种方式通用性更好,最近再使用 antd 开发的过程中,下载文件部分折腾了一下午,于是将关键的部分和遇到的一些问题整理如下。

前端核心代码

我的前端是基于 antd pro 开发的,这里不在详细介绍 antd pro 相关的内容,只说明下封装的下载函数:

import { request } from "umi";
// 这是我在项目中封装的下载函数,有2个参数:
// 一个是文件的id,用来给后端API搜索文件用的
// 一个是文件名filename,这个给前端用的,下载时,默认保存的文件名
export async function DownloadFile(id: string, filename: string) {
  return RestGet<BlobPart>(`/api/v1/file/download/${id}`, "blob").then(
    (res) => {
      let url = URL.createObjectURL(new Blob([res], { type: "octet/stream" }));
      let a = document.createElement("a");
      a.href = url;
      a.download = filename;
      a.click();
      URL.revokeObjectURL(url);
    }
  );
}
// GET 方式请求后端API
export async function RestGet<T>(
  url: string,
  responseType: "json" | "blob" = "json"
) {
  return request<T>(url, {
    method: "GET",
    headers: {
      "Content-Type": "application/json",
    },
    responseType: responseType,
  });
}
// POST 方式请求后端API
export async function RestPost<T>(
  url: string,
  body: any,
  responseType: "json" | "blob" = "json"
) {
  return request<T>(url, {
    method: "POST",
    headers: {
      "Content-Type": "application/json",
    },
    data: body,
    responseType: responseType,
  });
}

这个下载函数本质就是创建一个 <a> 元素,然后模拟点击此 <a> 元素来完成下载。
对于 DownloadFile函数,我是根据自己的需要封装的,不一定非得是这两个参数,可以根据自己的后端 API 来定制需要传入的参数。

将下载封装成函数之后看,前端就可以通过一个按钮来下载文件了,比如:

<Button
  type="link"
  key="download"
  icon={<DownloadOutlined />}
  onClick={() =>
    DownloadFile(item.video_file_id as string, (item.name as string) + ".mp4")
  }
>
  下载
</Button>

我这里是 GET 方式下载文件的,如果参数比较多且复杂的话,也可以使用 POST 方式。
只要把 DownloadFileRestGet<T>改成 RestPost<T>,并调整相应的参数即可。

遇到的问题

调试其中的下载方式就不提了,遇到的最大困难,搜索半天也没什么人提到的就是请求中的 responseType问题。

刚开始,我没有设置 responseType,默认值好像是 json或者 text
测试时发现,下载文本类型的文件没有问题,但是下载二进制文件的话,比如视频或者图片,下载之后总是无法正常打开,前后端也没有任何错误提示。

折腾了半天,才试出 responseType传入 blob的话,二进制文件才能正常下载。
而且发现,设置成 blob的话,文本类型的文件也能正常下载打开。

© 版权声明
好牛新坐标 广告
版权声明:
1、IT大王遵守相关法律法规,由于本站资源全部来源于网络程序/投稿,故资源量太大无法一一准确核实资源侵权的真实性;
2、出于传递信息之目的,故IT大王可能会误刊发损害或影响您的合法权益,请您积极与我们联系处理(所有内容不代表本站观点与立场);
3、因时间、精力有限,我们无法一一核实每一条消息的真实性,但我们会在发布之前尽最大努力来核实这些信息;
4、无论出于何种目的要求本站删除内容,您均需要提供根据国家版权局发布的示范格式
《要求删除或断开链接侵权网络内容的通知》:https://itdw.cn/ziliao/sfgs.pdf,
国家知识产权局《要求删除或断开链接侵权网络内容的通知》填写说明: http://www.ncac.gov.cn/chinacopyright/contents/12227/342400.shtml
未按照国家知识产权局格式通知一律不予处理;请按照此通知格式填写发至本站的邮箱 wl6@163.com

相关文章