来源 | https://www.cnblogs.com/wanglinmantan/p/15088419.html

前几天一个简单的下载图片的需求折腾了我后端大佬好几天,最终还是需要前端来搞,开始说不行的笔者最后又行了,所以趁着这个机会来总结一下。

先起个服务

使用expressjs起个简单的后端服务,先安装:

mkdir demo
cd demo
npm init
npm install express --save// v4.17.1

然后创建一个app.js文件,输入:

const express = require('express')
const app = express()
app.get('/', (req, res) => {res.send('hello world')
})
app.listen(3000, () => {console.log('服务启动完成')
})

然后在命令行输入:node app.js,访问http://localhost:3000/,页面显示hello world即表示服务启动成功。

接下来分别模拟几种情况:

情况1.静态图片

创建一个public文件夹,随便拷贝一张图片比如test.jpg进去,然后添加以下代码:

// ...
app.use(express.static('./public'))
// app.listen...

浏览器访问http://localhost:3000/test.jpg即可看到该图片。

情况2.读取图片文件,以流的形式返回

app.get('/getFileStream', (req, res) => {const fileName = req.query.nameconst stream = fs.createReadStream(path.resolve('./public/' + fileName))stream.pipe(res)
})

浏览器访问http://localhost:3000/getFileStream?name=test.jpg即可访问到该图片。

情况3.读取图片文件返回流并添加Content-Disposition响应头

Content-Disposition响应头是MIME协议的扩展,用来告诉浏览器如何处理服务器发送的文件,有三种取值:

Content-Disposition: inline// 如果浏览器能直接打开该文件会直接打开,否则触发保存
Content-Disposition: attachment// 告诉浏览器以附件的形式发送,会直接触发保存,会以接口的名字作为默认的文件名
Content-Disposition: attachment; filename="xxx.jpg"// 告诉浏览器以附件的形式发送,会直接触发保存,filename的值作为默认的文件名
app.get('/getAttachmentFileStream', (req, res) => {const fileName = req.query.name// attachment方法实际上设置了两个响应头的值:/*Content-Disposition: attachment; filename="【文件名】"Content-Type: 【文件MIME类型】*/res.attachment(fileName); const stream = fs.createReadStream(path.resolve('./public/' + fileName))stream.pipe(res)
})

情况4.动态生成图片返回流

我们以生成二维码为例,使用qr-image这个库来创建二维码,添加以下代码:

const qr = require('qr-image')
app.get('/createQrCode', (req, res) => {// 生成二维码只读流const data = qr.image(req.query.text, {type: 'png'});data.pipe(res)
})

情况5.返回base64字符串

app.get('/createBase64QrCode', (req, res) => {const data = qr.image(req.query.text, {type: 'png'});const chunks = []let size = 0data.on('data', (chunk) => {chunks.push(chunk)size += chunk.length})data.on('end', () => {const data = Buffer.concat(chunks, size)const base64 = `data:image/png;base64,` + data.toString('base64')res.send(base64)})
})

情况6.上述几种情况的post请求方式

// 解析json类型的请求体
app.use(express.json())
// 解析urlencoded类型的请求体
app.use(express.urlencoded())
app.post('/getFileStream', (req, res) => {const fileName = req.body.nameconst stream = fs.createReadStream(path.resolve('./public/' + fileName))stream.pipe(res)
})
app.post('/getAttachmentFileStream', (req, res) => {const fileName = req.body.nameres.attachment(fileName);const stream = fs.createReadStream(path.resolve('./public/' + fileName))stream.pipe(res)
})
app.post('/createQrCode', (req, res) => {const data = qr.image(req.body.text, {type: 'png'});data.pipe(res)
})

方法一、a标签下载

a标签html5版本新增了download属性,用来告诉浏览器下载该url,而不是导航到它,可以带属性值,用来作为保存文件时的文件名,尽管说有同源限制,但是我实际测试时非同源的也是可以下载的。

对于没有设置Content-Disposition响应头或者设置为inline的图片来说,因为图片对于浏览器来说是属于能打开的文件,所以并不会触发下载,而是直接打开,浏览器不能预览的文件无论有没有Content-Disposition头都会触发保存:

<!-- 直接打开 -->
<a href="/test.jpg" download="test.jpg" target="_blank">jpg静态资源</a>
<!-- 触发保存 -->
<a href="/test.zip" download="test.pdf" target="_blank">zip静态资源</a>
<!-- 触发保存 -->
<a href="https://www.7-zip.org/a/7z1900-x64.exe" download="test.zip" target="_blank">三方exe静态资源</a>
<!-- 直接打开 -->
<a href="/createQrCode?text=http://lxqnsys.com/" download target="_blank">二维码流</a>
<!-- 直接打开 -->
<a href="/getFileStream?name=test.jpg" download target="_blank">jpg流</a>
<!-- 触发保存 -->
<a href="/getFileStream?name=test.zip" download target="_blank">zip流</a>
<!-- 触发保存 -->
<a href="/getAttachmentFileStream?name=test.jpg" download target="_blank">附件jpg流</a>
<!-- 触发保存 -->
<a href="/getAttachmentFileStream?name=test.zip" download target="_blank">附件zip流</a>

所以说如果想用a标签下载图片,那么要让后端加上Content-Disposition响应头,另外也必须以流的形式返回,跨域图片符合这个要求也可以下载,即使响应没有允许跨域的头,但是静态图片即使添加了这个头也是直接打开:

// 经测试,浏览器仍然直接打开图片
app.use(express.static('./public', {setHeaders(res) {res.attachment()}
}))

和a标签方式类似的还可以使用location.href:

location.href = '/test.jpg'
location.href = '/test.zip'

行为和a标签完全一致。

这两种方式的缺点也很明显,一是不支持post等其他方式的请求,二是需要后端支持。

方法二、base64格式下载

a标签支持data:协议的URL,利用这个可以让后端返回base64格式的字符串,然后使用download属性进行下载:

<template><a :href="base64Img" download target="_blank">base64字符串</a>
</template>
<script>
import axios from 'axios'
export default {data () {return {base64Img: ''}},async created () {let { data } = await axios.get('/createBase64QrCode?text=http://lxqnsys.com/')this.base64Img = data}
}
</script>

这个方式就随便get还是post请求了,缺点是base64字符串可能会非常大,传输慢以及浪费流量,另外当然也得后端支持,需要同域或允许跨域。

方法三、blob格式下载

还是a标签,它还支持blob:协议的URL,利用这个可以把响应类型设置为blob,然后和base64一样扔给a标签:

<template><a :href="blobData" download target="_blank">blob</a>
</template>
<script>
import axios from 'axios'
export default {data () {return {blobData: null,blobDataName: ''}},async created () {let { data } = await axios.get('/test.jpg', {responseType: 'blob'})const blobData = URL.createObjectURL(data)this.blobData = blobData}
}
</script>

这个方式需要和上述几个需要通过ajax请求的一样,都需要后端可控,即图片同域或支持跨域。

方法四、使用canvas下载

这个方法其实和方法二和方法三是类似的,只是相当于把图片请求方式换了一下:

<template><a :href="canvasBase64Img" download target="_blank">canvas base64字符串</a><a :href="canvasBlobImg" download target="_blank">canvas blob</a>
</template><script>export default {data () {return {canvasBase64Img: '',canvasBlobImg: null}},created () {const img = new Image()// 跨域图片需要添加这个属性,否则画布被污染了无法导出图片img.setAttribute('crossOrigin', 'anonymous')img.onload = () => {let canvas = document.createElement('canvas')canvas.width = img.widthcanvas.height = img.heightlet ctx = canvas.getContext('2d')// 图片绘制到canvas里ctx.drawImage(img, 0, 0, img.width, img.height)// 1.data:协议let data = canvas.toDataURL()this.canvasBase64Img = data// 2.blob:协议canvas.toBlob((blob) => {const blobData = URL.createObjectURL(blob)this.canvasBlobImg = blobData})}img.src = '/createQrCode?text=http://lxqnsys.com/'}}
</script>

img标签是可以跨域的,但是跨域的图片绘制到canvas里后无法导出,浏览器会报错,可以给img添加crossOrigin属性,但是,如果图片没有允许跨域的头加了也没用。

方法五、表单形式下载

对于post请求方式下载图片的话,除了使用上述的方法二和方法三之外,还可以使用form表单:

<template><el-button type="primary" @click="formType">from表单下载</el-button></div>
</template><script>
export default {methods: {formType () {// 创建一个隐藏的表单const form = document.createElement('form')form.style.display = 'none'form.action = '/getAttachmentFileStream'// 发送post请求form.method = 'post'form.target = '_blank'document.body.appendChild(form)const params = {name: 'test.jpg'}// 创建input来传递参数for (let key in params) {let input = document.createElement('input')input.type = 'hidden'input.name = keyinput.value = params[key]form.appendChild(input)}form.submit()form.remove()}}
}
</script>

使用该方式,图片流的响应头需要设置Content-Disposition,否则浏览器也是直接打开图片,有该响应头的话跨域图片也可以下载,即使图片不允许跨域。

方法六、ifrmae下载

document.execCommand有一个SaveAs命令,可以触发浏览器的另存为行为,利用这个可以把图片加载到iframe里,然后通过iframe的document来触发该命令:

<template><el-button type="primary" @click="iframeType">iframe下载</el-button>
</template><script>export default {methods: {iframeType () {const iframe = document.createElement('iframe')iframe.style.display = 'none'iframe.onload = () => {iframe.contentWindow.document.execCommand('SaveAs')document.body.removeChild(iframe)}iframe.src = '/createQrCode?text=http://lxqnsys.com/'document.body.appendChild(iframe)}}}
</script>

图片必须要是同源的,这种方式了解一下就行,因为它只在IE里被支持。

小结

本文简单分析了一下前端下载图片的各种方式,各位可以根据实际需求进行选择,除了最后一种方法,其余方法均未在IE上测试,有需要的可以自行测试。

demo代码地址:https://github.com/wanglin2/download-image-demo

感谢阅读。

学习更多技能

请点击下方公众号

6种实现前端下载图片的方法汇总相关推荐

  1. 前端下载图片的N种方法

    前几天一个简单的下载图片的需求折腾了我后端大佬好几天,最终还是需要前端来搞,开始说不行的笔者最后又行了,所以趁着这个机会来总结一下下载图片到底有多少种方法. 先起个服务 使用expressjs起个简单 ...

  2. HTML+JS实现浏览器前端下载图片

    最近做项目需要用到js图片下载,按照原先下载文件的思路,我感觉直接 window.open("下载链接"); 就可以了,但实际操作后,发现这个只会在新窗口打开图片,并不会触发下载, ...

  3. php curl 下载图片 花,PHP curl下载图片的方法

    PHP curl下载图片的方法 $images = [ 'http://wx.qlogo.cn/mmopen/vi_32/Q0j4TwGTfTKPkia3rxjuBoSQ1sUSmqYXDazibuI ...

  4. python requests下载图片_Python使用urllib库、requests库下载图片的方法比较

    我们知道,使用Python制作一个爬虫程序,requests库是一个不错的选择,requests发送网络请求非常简单,比较使用urllib库,使用requests库可以大大减少代码量,它还可以定制请求 ...

  5. JavaScript 实现前端下载图片

    JavaScript 实现前端下载图片 前几天公司的项目有个需求,实现点击下载图片功能,当时我就想着还不简单嘛,当即写下以下代码: const download = document.createEl ...

  6. 前端下载图片、pdf、excel、world文件;前端下载图片和pdf文件;前端a标签下载图片和pdf文件;下载文件名称不生效原因。

    参考文章1 参考文章2 以下的下载是,拿到了后端给的下载图片的接口地址url > 方法1:将文本或者JS字符串通过 Blob 转换成二进制下载 优点:可以下载,也可以保存名称. //文件流参数和 ...

  7. 如何利用后端给的url,前端下载图片、音频、视频等文件

    后端给了url,让前端处理下载,刚开始只处理图片url下载,然后用最简单的方式,创建a标签,叫download属性,发现竟然不行,找了一下原因,原来是url地址跨域了,这里需要让后端处理一下服务器对u ...

  8. php实现云盘下载不限速,【合集】【已更新第五种】五种百度云下载不限速方法+软件...

    本帖最后由 kof20046 于 2019-3-17 17:46 编辑 目前本人知道的五种百度网盘不限速方法,给大家收集起来,方便查找下载. 还有IDM和FDM也是可以下载,但是我下载的时候总只有10 ...

  9. python根据url下载图片的方法

    1.使用urllib.request.urlretrieve()方法 from urllib import requestrequest.urlretrieve(url, path + '.jpg') ...

最新文章

  1. HDU 2191 悼念512汶川大地震遇难同胞——珍惜现在,感恩生活(多重背包)
  2. Java任务调度框架Quartz
  3. Java 集合类(一)
  4. 手游极品飞车无限狂飙链接服务器失败,极品飞车无极限无法联网是什么原因 联网失败原因分析及解决方法...
  5. Kaggle Bike Sharing Demand Prediction – How I got in top 5 percentile of participants?
  6. 使用接口更改已装箱的值类型中的字段
  7. odoo16 Windows绿色版 下载就能尝试了
  8. 高德地图ios11 定位失败
  9. 1092 最好吃的月饼
  10. MarkDown编辑器设置图片大小
  11. 手把手教你如何PCB板材选型(二)
  12. 2023年春节放假时间安排计划用哪一提醒备忘软件记录?
  13. Photoshop鼠绘教程:逼真鲜活金鱼的绘制
  14. wstmart商城系统研究日志一
  15. 码距和检错纠错能力的关系、CRC和Hamming Code
  16. 杨昕立计算机学院,写在告别之前——那些来自辅导员们的悄悄话
  17. Git是什么?有什么用?
  18. FC按键修改教程之一键开关
  19. [转贴]去除迅雷广告和弹出广告窗口
  20. 你必须知道的互联网协议详解

热门文章

  1. 【数据分析】APP活动运营:应关注哪些数据指标(in)
  2. ACM经历和退役感想
  3. java修改图片宽高
  4. Instance Segmentation 和 Semantic Segmentation
  5. matlab扩频仿真,#基于matlab的直接序列扩频通信系统仿真
  6. C语言跳过障碍物小游戏,html5飞翔的小鸟越过障碍物小游戏代码
  7. vim退出快捷键_vim常用快捷键及其功能
  8. c 语言cad 二次开发,cad编程语言(AUTO CAD 得二次开发语言及工具)
  9. [NTT][DP][树链剖分][分治] LOJ #6289. 花朵
  10. 施耐德服务器显示DIS,施耐德官方培训05-过电压防护及电涌保护器的选择应用_V1.ppt...