Skip to content

下载文件

location.href / window.open

适用:

  • get 请求
  • 单文件下载
js
window.location.href = url

链接地址下载

js
const link = document.createElement('a')
const url = `url下载链接地址`
link.href = url
link.setAttribute('download', '文件名') // 同域才生效
document.body.appendChild(link)
link.click()
document.body.removeChild(link)
html
<a href="url下载链接地址" download="文件名"></a>

文件流下载

服务端代码 以 Nodejs 的 fastify 框架为例:

js
const fastify = require('fastify')({ logger: true })
const cors = require('@fastify/cors')
const fs = require('fs')
const path = require('path')

fastify.register(cors, {
  // put your options here
})

// 设置允许下载的文件所在的目录
const downloadDir = path.join(__dirname, 'downloads')

fastify.post('/download/:filename', async (request, reply) => {
  const { filename } = request.params
  const filePath = path.join(downloadDir, filename)

  try {
    // 检查文件是否存在
    await fs.promises.access(filePath, fs.constants.F_OK)

    reply.header('access-control-expose-headers', 'Content-Disposition')
    reply.header('Content-Disposition', `attachment; filename=${filename}`)
    reply.type('application/octet-stream')

    // 创建文件读取流并将其通过管道传输到响应中
    const fileStream = fs.createReadStream(filePath)
    return reply.send(fileStream)
  } catch (error) {
    reply.code(404).send({ error: '文件不存在' })
  }
})

const start = async () => {
  try {
    await fastify.listen({ port: 3000 })
    console.log(`服务器运行在 http://localhost:3000`)
  } catch (err) {
    fastify.log.error(err)
    process.exit(1)
  }
}

start()

⚠️ 注意:

允许浏览器访问 Content-Disposition 头,在 CORS 默认情况下只暴露出了 CORS-safelisted response header 包括:

  • Cache - Control
  • Content - Language
  • Content - Length
  • Content - Type
  • Expires
  • Last - Modified
  • Pragma

通过 Access-Control-Expose-Headers 添加期望暴露的其他请求头。

客户端代码

js
const req = new XMLHttpRequest()
req.open('POST', 'http://localhost:3000/download/123.txt', true)
req.responseType = 'blob'
req.setRequestHeader('Content-Type', 'application/json')
req.onload = function () {
  const content = req.getResponseHeader('Content-Disposition')
  // 文件名最好用后端返的Content-disposition
  // 需要后端设置 Access-Control-Expose-Headers: Content-disposition 使得浏览器将该字段暴露给前端
  // 获取文件名的处理需要看后端的 Content-Disposition 字段如何设置的
  const name = content && content.split('filename=')[1]
  const fileName = decodeURIComponent(name)

  const link = document.createElement('a')
  link.href = window.URL.createObjectURL(new Blob([req.response])) // 创建下载的链接
  link.download = fileName || '默认文件名' // 下载后文件名
  link.style.display = 'none'
  document.body.appendChild(link)
  link.click() // 点击下载
  window.URL.revokeObjectURL(link.href) // 释放掉blob对象
  document.body.removeChild(link) // 下载完成移除元素
}
req.send(JSON.stringify(params))
js
axios({
  method: 'post',
  url: 'http://localhost:3000/download/123.txt',
  responseType: 'blob',
})
  .then(function (resp) {
    const fileName = resp.headers['content-disposition'].split('filename=')[1]

    const link = document.createElement('a')
    link.href = window.URL.createObjectURL(new Blob([resp.data])) // 创建下载的链接
    link.download = fileName || '默认文件名' // 下载后文件名
    link.style.display = 'none'
    document.body.appendChild(link)
    link.click() // 点击下载
    window.URL.revokeObjectURL(link.href) // 释放掉blob对象
    document.body.removeChild(link) // 下载完成移除元素
  })
  .catch((e) => {
    console.log(e)
  })