前端开发中文件下载和Base64位转化

一、前言

这两天在开发过程中遇到下载图片的需求,没想到常规的写法无法下载,只能在浏览器打开,之前还真没没注意过,思考下最近做的关于文件操作的内容,做下总结。

  • 下载媒体资源
  • 上传媒体(图片、音频、视频)到云存储
  • 本地解析媒体资源
  • Base64位转码给后端保存

二、关于前端文件下载

前端几种常见的下载文件的方法

  • 1、a标签
  • 2、js
  • 3、window.open

需要注意是的,a标签对于跨域的资源是有不同表现的。

  • 当下载资源和当前资源跨域,且当前资源可以直接在浏览器打开预览(比如图片、PDF)将无法下载,会在浏览器打开,解决方法在后面
  • 当下载资源和当前资源跨域,且资源浏览器无法预览(比如XLSX)将会下载该文件
  • 不跨域的情况下将会下载

1、a标签

大部分文件都可以用a标签+download属性来下载,简单方便。

html 复制代码
<a href="path/to/file.pdf" download="filename.pdf">点击下载文件</a>

2、js下载文件

js下载文件也是通过创建a标签来实现的,对比a标签的下载更灵活。

js 复制代码
export const downloadFile = ({
    fileUrl, 
    fileName, 
    callback
}:{
    fileUrl: string,
    fileName: string,
    callback?: (error?:any) => void
}) => {
  try {
    const url = fileUrl;
    const link = document.createElement('a');
    link.target = '_blank';
    link.style.display = 'none';
    link.href = url;
    link.setAttribute('download', fileName);
    document.body.appendChild(link);
    link.click();
    document.body.removeChild(link);
    callback && callback();
  } catch (err) {
    callback && callback(err);
  }
};

因为JS也是通过创建A标签来实现的,所以需要先拉取Blob格式的资源,然后创建本地域名一致的资源URL,再进行下载

js 复制代码
export const downloadCorsFile = ({
    fileUrl, 
    fileName, 
    callback
}:{
    fileUrl: string,
    fileName: string,
    callback?: (error?:any) => void
}) => {
  try {
    // 也可以使用fetch拉取资源,不过要注意兼容性
    const xhr = new XMLHttpRequest();
    xhr.open('get', fileUrl, true);
    xhr.responseType = 'blob';
    xhr.send();
    xhr.onload = function (res) {
      const url = window.URL.createObjectURL(res.target.response);
      downloadFile({url, fileName, callback});
    };
  } catch (err) {
    callback(err);
  }
};

3、利用浏览器行为下载

window.openwindow.location.href是利用浏览器对应的URL就会将文件下载下来。可以用来下载XLSX、ZIP文件等。

js 复制代码
window.open('https://demo/download_filename.xlsx', '_blank')
window.location.href = 'https://demo/download_filename.xlsx'

三、文件转化成Base64位

1、File或Blob类型转化成base64位

在开发中,我们常常要将文件转化成base64位,文件可能是手动上传的文件、通过URL下载的图片等。

原理都是通过FileReader对象,调用实例方法readAsDataURL将文件读取为Base64位。

js 复制代码
const reader = new FileReader();
reader.readAsDataURL(file);
reader.onloadend = () => {
  const result = reader.result;
}

由于异步获取的base64位,为了方便使用,我们常常通过Promise处理下

js 复制代码
// File/Blob转化成base64位
export function fileToBase64(file: File | Blob): Promise<string> {
  return new Promise((resolve, reject) => {
    const reader = new FileReader();
    reader.readAsDataURL(file);
    reader.onloadend = () => {
      const result = reader.result;
      resolve(result);
    };
    reader.onerror = reject;
  });
}

2、通过URL获取文件的类型,然后将Blob转化成base64位

js 复制代码
export function urlToBase64(url) {
  return fetch(url)
    .then((response) => response.blob())
    .then((blob) => {
      return fileToBase64(blob)
    });
}

FileReader对象的几种常见方法包括:

  • readAsBinaryString(Blob or File):将输入的 Blob 或 File 对象读取为二进制字符串。
  • readAsArrayBuffer(Blob or File):将输入的 Blob 或 File 对象读取为一个 ArrayBuffer。
  • readAsDataURL(Blob or File):将输入的 Blob 或 File 对象读取为一个 Base64 编码的字符串。
  • readAsText(Blob or File, encoding):将输入的 Blob 或 File 对象读取为一个文本字符串,其中第二个参数指定字符编码。
相关推荐
Zoey的笔记本21 分钟前
敏捷与稳定并行:Scrum看板+BPM工具选型指南
大数据·前端·数据库·python·低代码
文心快码BaiduComate23 分钟前
0代码手写!体验百度Comate的“魔法”:我造了个会理解情绪的中介层
前端·程序员·前端框架
38242782730 分钟前
表单提交验证:onsubmit与return详解
前端·javascript·html
前端小蜗32 分钟前
普通前端程序员的 2025:没什么大胜利,但也没被生活击倒
前端
bug总结1 小时前
身份证号脱敏的正确实现
前端·javascript·vue.js
林太白1 小时前
Vite8 Beta来了,Rolldown携手Oxc
前端·javascript·后端
xkxnq1 小时前
第二阶段:Vue 组件化开发(第 19天)
前端·javascript·vue.js
技术净胜2 小时前
Python 操作 Cookie 完全指南,爬虫与 Web 开发实战
前端·爬虫·python
神奇的程序员2 小时前
Nginx日志分析工具-NginxPulse开源了
前端
我是小疯子662 小时前
前端开发入门:HTML、CSS与JS学习指南
前端