首页 > 作文

浅析HTML5中的download属性使用

更新时间:2023-04-03 03:36:04 阅读: 评论:0

随着前端技术的发展,越来越多的业务场景中需要前端来处理文件下载。在众多的方法中,通过 <a> 标签的 download 属性实现下载是其中常见也是比较简单的一种方法。

download 属性介绍

常规的 <a> 标签通过 href 实现链接跳转,如果只想下载文件而不是跳转预览,最好的方式是在 <a> 标签中添加 download 属性,就能很简单地实现下载操作。

download 是 html5 中 <a> 标签新增的一个属性,此属性会强制触发下载操作,指示浏览器下载 url 而不是导航到它,并提示用户将其保存为本地文件,例如:

<a href="result.png" download>download</a>

如果缺少 download 属性,点击 “download” 会直接变成预览图片,当添加 download 属性后则会触发图片的下载。

目前 download 属性的兼容性如caniu 中所展示的:

可以以看到,大部分主流的浏览器基本都已经支持 download 属性,而 ie 的表现一如既往的感人,目前许多 window 系统仍然在使用 ie ,这也是许多业务需求需要考虑的。这种兼容性问题限制了 download 的更广泛应用。

动态资源下载

面对一些动态内容下载的业务场景,即图片等资源的地址并不是固定的(例如一些在线绘图工具所生成的图片),只使用 html 无法满足需求。为了能够满足不同的 url 下载,可以通过js 实现一个动态触发 url 下载的方法。

function download(href, filename='')  {  const a = document.createelement('a')  a.download = filename  a.href = href  document.body.appendchild(a)    a.click()  a.remove()}

需要注意的是,代码中对创建的 <a> 进行的 appendchildremove 操作主要是为了兼容 firefox 浏览器,在 firefox 浏览器下调用该方法如果不将创建的 <a> 标签添加到 body 里,点击链接不会有任何反应,无法触发下载,而在 chrome 浏览器中则不受此影响。

上述的方法可以实现同源资源的下载。但在很多场景中,还需要处理跨域资源。遗憾的是, download 属性目前仅适用于 同源 url ,即如果需要下载的资源地址是跨域的, download 属性就会失效,点击链接会变成导航预览。

测试:点击下载,结果只是预览而无法下载图片。

注: chrome65 之前是支持 download 属性触发文件跨域下载的,之后则严格遵循同源策略,无法再通过 download 属性触发跨域资源的下载。而 firefox 一直不支持跨域资源的 download 属性下载。

文件命名问题

download 属性不仅可以触发下载,也能指定下载文件名:

<a href="test.png" download="joker.png">下载</a>

如果下载文件的后缀与源文件保持一致,可以设置缺省文件名:

<a href="test.png" download="joker">下载</a>

笔者曾遇到过一个问题,通过 <a> 标签触发跨域资源下载,代码与上述的 download 方法基本相同,只是在设置 download 属性的地方有点不同:

a.tattribute(download, true)

结果在老版本的 chrome 浏览器中出现了如下情况。

下载文件名成了 true 。很明显,浏览器将 download 属性值读成了文件名。爱国的古诗

经过分析,出现上述问题主要是因为:

1. 首先本不该将 download 设为 truedownloaddisabled 这种类型的属性值不同,它与文件名直接相关联。而且对于这种前后端响应式下载的方式, download 属性并不是必要的。

2. 在 chrome 的早期版本不仅支持跨域资源的 download 属性下载,而且还可以通过 download 重置跨域资源的文件名,因此才会出现上述这种情况。

前后端配合完成文件下载的业务场景,一般是由后端设置响应头中的 c忻州旅游景点大全ontent-disposition 信息来实现。

在 http 场景中,contexp win7nt-disposition 第一个参数或者是 inline(默认值,表示回复中的消息体会以页面的一部分或者整个页面的形式展示),或者是 attachment(意味着消息体应该被下载到本地;大多数浏览器会呈现一个“保存为”的对话框,将 filename 的值预填为下载后的文件名)。

如果在响应头中设置了 content-disposition ,前端也在对应链接的 <a> 标签中添加了 download 属性,那么此时命名规则:

如果 http 头中的 content-disposition 属性赋予了一个不同于此属性的文件名,http 头属性优先于此属性。

经过测试发现,当 http 头中 content-disposition 不为空时:

在 chrome 浏览器中,不管 http 头中 content-disposition 的第一个参数被设为 attachment 还是 inline ,只要设置了 filename, download 就无法重置文件名。相反,当 filename 为空时, download 属性值会被设为文件名。 在 firefox 浏览器中,浏览器只会读取 content-disposition 的 filename 值,若是filename 为空,则取源文件名。此时 download 无论如何都无法重置文件名。

总结一下: 未在响应头设置 content-disposition 信息(例如一般直接定位资源的同源url), download 属性可以重置文件名。若后端在 content-disposition 字段中已经设置了 filename,以 filename 值为准。

对于后端已经设定了文件名的情况下,如果仍然想要对文件名进行重置,该如何处理呢?

blob: url

关于 download 属性还有介绍:

尽管 http url 需要位于同一源中,但是可以使用 blob: url 和 data: url ,以方便用户下载使用 javascript 生成的内容(例如使用在线绘图 web 应用程序创建的照片)。

blob (binary large object)即二进制大对象,这个我们并不陌生,一些数据库将blob用来表示存储二进制文件的字段类型。file 接口也是基于 blob,继承了 blob 的功能并将其扩展使其支持用户系统上的文件。blob 对象通过 blob 构造函数来创建:

blob(blobparts[, options])

var debug = {hello: "world"};var blob = new blob([json.stringify(debug, null, 2)],  {type : 'application/json'});

如果需要实现一些简单的文本或 js 字符串之类的文件下载,可以通过将文本信息转成 blob 二进制流,生成一个 blob url,配合 download 属性完成下载,代码如下。

const downloadtext = (text, filename = '') {  const a = document.createelement('a')  a.download = filename  const blob = new blob([text], {type: 'text/pla原子核的能级in'})    // text指需要下载的文本或字符串内容  a.href = window.url.c洋湖湿地公园reateobjecturl(blob)   // 会生成一个类似blob:http://localhost:8080/d3958f5c-0777-0845-9dcf-2cb28783acaf 这样的url字符串  document.body.appendchild(a)    a.click()  a.remove()}

这种 blob url 与常见的 http url 有什么区别呢?

blob url / object url是一种伪协议,可以让blob和file对象用作图像和二进制数据下载链接等url源。

浏览器在内部通过 url.createobjecturl() 创建一个对 blob 或 file 对象的特殊引用,生成的 blob url 只能在浏览器本地的单个实例和同一会话中使用,并且这个 url 对象会在页面退出的时候被浏览器释放。

因此 blob url 并不能指向一个服务器资源 ,你无法在其它页面中打开它。同时由于编码格式有所差别,blob url 比起 data urls 所占的空间资源更少,性能也更好。

blob 为 web 开发提供了一个非常有用的功能:创建 blob url。将二进制数据封装为 blob 对象,然后使用 url.createobjecturl() 生成 blob url,由于blob url本身就是一个同源url,可以使用该 url 配合 download 解决跨域资源的下载以及命名问题。

解决方案

通过 blob 和 fetch 可以解决跨域和文件命名的问题:使用 fetch 获取跨域资源返回一个blob 对象并生成一个 blob url,配合 <a> 标签的 download 属性触发下载,代码如下:

function download(href, filename = '')  {  const a = document.createelement('a')  a.download = filename  a.href = href  document.body.appendchild(a)    a.click()  a.remove()}function downloadfile(url, filename='') {  fetch(url, {    headers: new headers({      origin: location.origin,    }),    mode: 'cors',  })    .then(res => res.blob())    .then(blob => {      const bloburl = window.url.createobjecturl(blob)      download(bloburl, filename)      window.url.revokeobjecturl(bloburl)    })}

此时再点击下载,可以正常的将跨域图片下载到本地了。

需注意的是跨域资源所在的服务器需要配置 access-control-allow-origin 信息,否则会得到一个大写的跨域报错。header 配置例如:

// 允许任何域名访问header('access-control-allow-origin: *');//指定域名访问header('access-control-allow-origin: https://h5.ele.me');

目前这种方法还存在一些不足,例如浏览器会限制 blob 数据大小不超过500m,在性能方面也会有所不足。

总结

目前前端有很多种下载方法, download 属性下载属于其中比较简单的一种,不过仔细考量其中的一些特性也能挖掘出很多有用的信息。 download 与浏览器特性紧密相关,目前该属性的兼容性也是一大问题,不过连微软官方都恳求用户不要再使用 ie ,相信以后 download 的兼容性问题会持续得到改善,应用前景也会越来越广阔。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持www.887551.com。

本文发布于:2023-04-03 03:35:59,感谢您对本站的认可!

本文链接:https://www.wtabcd.cn/fanwen/zuowen/cc9c00c15b4edddfeb8109c926ec3803.html

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。

本文word下载地址:浅析HTML5中的download属性使用.doc

本文 PDF 下载地址:浅析HTML5中的download属性使用.pdf

标签:属性   文件名   资源   浏览器
相关文章
留言与评论(共有 0 条评论)
   
验证码:
Copyright ©2019-2022 Comsenz Inc.Powered by © 专利检索| 网站地图