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

先起个服务

使用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.name
const 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 = 0
data.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.name
const stream = fs.createReadStream(path.resolve('./public/' + fileName))
stream.pipe(res)
})
app.post('/getAttachmentFileStream', (req, res) => {
const fileName = req.body.name
res.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.width
canvas.height = img.height
let 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 = key
input.value = params[key]
form.appendChild(input)
}
form.submit()
form.remove()
}
}
}
</script>

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

六.ifrmae下载

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

<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

前端下载图片的N种方法的更多相关文章

  1. python批量下载图片的三种方法

    一是用微软提供的扩展库win32com来操作IE: win32com可以获得类似js里面的document对象,但貌似是只读的(文档都没找到). 二是用selenium的webdriver: sele ...

  2. Python 下载图片的几种方法

    import osos.makedirs('./image/', exist_ok=True)IMAGE_URL = "http://image.nationalgeographic.com ...

  3. Python 下载图片的三种方法

    import os os.makedirs('./image/', exist_ok=True) IMAGE_URL = "http://image.nationalgeographic.c ...

  4. Python下载网页的几种方法

    get和post方式总结 get方式:以URL字串本身传递数据参数,在服务器端可以从'QUERY_STRING'这个变量中直接读取,效率较高,但缺乏安全性,也无法来处理复杂的数据(只能是字符串,比如在 ...

  5. DISCUZ论坛添加页头及页尾背景图片的几种方法

    先给大家分享页头添加背景图片的两种方法:1. 第一种效果,是给discuz的整体框架添加背景图片,见图示: 添加方法如下:找到你现在使用模板common文件下的header.html文件,在<h ...

  6. php 下载远程图片 的几种方法(转)

    1.获取远程文件大小及信息的函数 function getFileSize($url){          $url = parse_url($url);          if($fp = @fso ...

  7. UIImage加载图片的两种方法区别

    Apple官方的文档为生成一个UIImage对象提供了两种方法加载图片: 1. imageNamed,其参数为图片的名字: 2. imageWithContentsOfFile,其参数也是图片文件的路 ...

  8. AE 将地图导出为图片的两种方法

    在ArcGIS的开发中,我们经常需要将当前地图打印(或是转出)到图片文件中.将Map或Layout中的图象转出有两种方法,一种为通过IActiveView的OutPut函数,另外一种是通过IExpor ...

  9. Android TextView里直接显示图片的三种方法

    方法一:重写TextView的onDraw方法,也挺直观就是不太好控制显示完图片后再显示字体所占空间的位置关系.一般假设字体是在图片上重叠的推荐这样写.时间关系,这个不付源代码了. 方法二:利用Tex ...

随机推荐

  1. toFixed()与银行家舍入

    toFixed()与银行家舍入 一直在用toFixed()方法做浮点数的舍入取值,如果只是客户端展示数据是没有多大问题的,但是如果涉及到和后端互交,数据的精度可能会导致接口对接失败,当然了,涉及安全性 ...

  2. 2021.08.10 Euler函数总结

    2021.08.10 Euler函数总结 知识: 记 φ(n) 表示在 [1,n] 中与 n互质的数的个数. 1.p为质数,则 \[φ(p^l)=p^l-p=p^{l-1}(p-1) \] 注:每p个 ...

  3. Math内置对象 常用的方法

    属性: Math.Pi 方法: Math.max()   最大值 Math.min()  最小值 Math.ceil()  向上取整 Math.floor() 向下取整 Math.random()   ...

  4. 漏洞复现:MS10-046漏洞

    漏洞复现:MS10-046漏洞 实验工具1.VMware虚拟机2.Windows7系统虚拟机3.Kali 2021 系统虚拟机 1.在VMware中打开Windows7虚拟机和Kali 2021虚拟机 ...

  5. 不使用比较和条件判断实现min函数的一种方法

    不使用比较和条件判断实现min函数,参数为两个32位无符号int. 面试的时候遇到的题目,感觉很有意思. 搜了一下多数现有的解法都是仅有两种限制之一,即要么仅要求不能使用比较,要么仅要求不能使用条件判 ...

  6. 为什么不建议给MySQL设置Null值?《死磕MySQL系列 十八》

    大家好,我是咔咔 不期速成,日拱一卒 之前ElasticSearch系列文章中提到了如何处理空值,若为Null则会直接报错,因为在ElasticSearch中当字段值为null时.空数组.null值数 ...

  7. 一条Sql的执行过程

    一条sql内部是如何执行的: 学习MySQL实战45专栏 sql中的内部执行图: 可以分为两部分:server和存储引擎 server层包含: 连接器.分析器.优化器.执行器,涵盖了MySQL大多数核 ...

  8. 使用Spring MVC开发RESTful API

    第3章 使用Spring MVC开发RESTful API Restful简介 第一印象 左侧是传统写法,右侧是RESTful写法 用url描述资源,而不是行为 用http方法描述行为,使用http状 ...

  9. Primal_Dual 原始对偶

    不是费用流都需要用 SPFA 吗. 众所周知,SPFA 去世了,然后网络流显然有负边.于是我们可以像 Johnson 全源最短路一样,给边加上势能,具体实现看我之前的 博客 啦. 然后对于每一次跑 D ...

  10. MySQL中读页缓冲区buffer pool

    Buffer pool 我们都知道我们读取页面是需要将其从磁盘中读到内存中,然后等待CPU对数据进行处理.我们直到从磁盘中读取数据到内存的过程是十分慢的,所以我们读取的页面需要将其缓存起来,所以MyS ...