前端导出 Word/Excel/PDF 文件

核心思路是:后端返回文件二进制流 → 前端接收并转换为 Blob → 创建下载链接触发保存

一、核心实现步骤(通用逻辑)

  1. 后端接口需返回 文件二进制流(Content-Type 对应文件类型),而非 JSON。
  2. 前端请求时设置 responseType: 'blob',确保接收二进制数据。
  3. 将 Blob 转换为临时下载链接,模拟点击实现文件保存。
  4. 清理临时链接,避免内存泄漏。

二、完整代码实现(Axios 版本,最常用)

1. 导出 Excel 文件(.xlsx)

javascript 复制代码
// 安装依赖:npm install axios
import axios from 'axios';

/**
 * 导出Excel文件
 * @param {string} apiUrl - 接口地址
 * @param {object} params - 接口参数(如筛选条件)
 * @param {string} fileName - 自定义文件名(无需后缀)
 */
export const exportExcel = async (apiUrl, params, fileName = '导出数据') => {
  try {
    // 1. 发送请求,指定接收二进制流
    const response = await axios({
      url: apiUrl,
      method: 'GET', // 也可POST,根据后端接口调整
      params: params, // POST请用data: params
      responseType: 'blob', // 关键:指定返回Blob类型
      headers: {
        'Content-Type': 'application/json', // 根据后端要求调整
        // 如有token,添加认证:
        // 'Authorization': `Bearer ${localStorage.getItem('token')}`
      }
    });

    // 2. 处理返回的Blob数据
    const blob = new Blob([response.data], {
      type: 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet' // Excel MIME类型
    });

    // 3. 创建临时下载链接
    const downloadUrl = window.URL.createObjectURL(blob);
    const link = document.createElement('a');
    link.href = downloadUrl;
    link.download = `${fileName}.xlsx`; // 文件名+后缀
    document.body.appendChild(link);

    // 4. 触发下载
    link.click();

    // 5. 清理资源
    document.body.removeChild(link);
    window.URL.revokeObjectURL(downloadUrl);

    console.log('Excel文件导出成功');
  } catch (error) {
    console.error('Excel导出失败:', error);
    alert('文件导出失败,请重试');
  }
};

// 调用示例
// exportExcel('/api/export/user', { department: '技术部' }, '技术部用户列表');

2. Blob的type值

  • 如果你的接口返回的是 .xls 格式(旧版二进制),就用:

    go 复制代码
    const type = 'application/vnd.ms-excel';
    const blob = new Blob([response.data], { type: type });

    下载时文件名后缀用 .xls

  • 如果接口返回的是 .xlsx 格式(新版 XML),就用:

    go 复制代码
    const type = 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet';
    const blob = new Blob([response.data], { type: type });

    下载时文件名后缀用 .xlsx

⚠️ 注意:MIME 类型必须和实际文件格式一致,否则可能导致下载的文件无法打开。

3. 导出 Word 文件(.docx)

仅需修改 Blob 的 type 和文件名后缀,核心逻辑完全一致:

go 复制代码
// 新版 文件格式.docx
    const blob = new Blob([response.data], {
      type: 'application/vnd.openxmlformats-officedocument.wordprocessingml.document' // Word MIME类型
    });
    
// 旧版 文件格式.doc
const blob = new Blob([response.data], {
      type: 'application/msword' 
    });

  

4. 导出 PDF 文件(.pdf)

go 复制代码
    // 创建 Blob,指定 PDF 类型
    const blob = new Blob([response.data], {
      type: 'application/pdf'
    });
相关推荐
bluceli2 小时前
JavaScript异步编程深度解析:从回调到Async Await的演进之路
前端·javascript
青青家的小灰灰2 小时前
Vue 3 新标准:<script setup> 核心特性、宏命令与避坑指南
前端·vue.js·面试
SuperEugene2 小时前
路由与布局骨架篇:布局系统 | 头部、侧边栏、内容区、面包屑的拆分与复用
前端·javascript·vue.js
代码煮茶2 小时前
前端网络请求实战 | Axios 从入门到封装(拦截器 / 错误处理 / 重试)
javascript
进击的尘埃2 小时前
组合式函数 Composables 的设计模式:如何写出可复用的 Vue3 Hooks
javascript
大金乄2 小时前
用canvans画一个流程图
前端
大金乄2 小时前
TreeSelect 是一个基于 Element UI 的树形选择器组件,结合了 el-select 和 el-tree 的功能,支持单选和多选模式,支持树形
前端
大金乄2 小时前
自动构建打包脚本(开发环境)
前端
进击的尘埃2 小时前
浏览器渲染管线深度拆解:从 Parse HTML 到 Composite Layers 的每一帧发生了什么
javascript