前端文件下载通识篇

前言

前端如何实现下载文件呢?随着前端技术的发展,越来越多的前端需求中会出现下载文件这样的需求。

看着掘金很多人在近期不断的分享有关的文章,我总结了下自己的经验,根据不同情况,总结了一篇算是前端文件下载的通识篇,如果你对这方面完全不懂或者没有任何方案,那么本文会给你一个很不错的启示。

方案一 :原生提交,后端返回文件流

这种方式是利用form.submit直接向后端提交,后端返回文件流生成的文件,后端处理成功后会直接返回到页面,浏览器会整理并打开自己的保存下载文件机制 。

优点 :没有兼容问题,传统方式

缺点:拿不到后端处理这个过程的时机,无法根据回调函数做交互以及进度提示

// 后端参考代码
return File(stream, "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet", HttpUtility.UrlEncode("咨询记录导出.xls", Encoding.GetEncoding("UTF-8")));
// 参考代码
function exportRecord() {
   var $form = $("<form>"); //定义一个form表单
    $form.hide().attr({target:'',method:'post','action':'/xxx'});
    var $input = $("<input>");
    $input.attr({"type":"hidden","name":'req'}).val(req);
    $form.append($input).appendTo($("body")).submit().remove();
 }

方案 二 :ajax提交,后端返回在线文件地址

利用ajax或者新生的axios去提交请求,后端会返回一个线上的文件地址,前端可以通过原生的window.open打开这个地址就可以实现下载;也可以通过a标签设置href以及download属性,并自动点击实现其下载功能,关于其兼容性问题,可以判断download属性是否存在来弥补。

优点 :可以拿到其返回时机,可以做交互

缺点 :线上会存储大量的中间临时文件,可以用设置时限来优化。另外涉及用户隐私的问题,可以用token等验证机制实现。

// 参考方案
$.ajax({
                type: "post",
                url: "/xxx",
                data: data,
                success: function (res) {
                    tool.loadingend();
                    if(res.Status){
                    // window.open或者a标签下载 
                    var isSupportDownload = 'download' in document.createElement('a');
                    if(isSupportDownload){
                    var $a = $("<a>") ;
                    $a.attr({href:res.url,download:'filename'}).hide().appendTo($("body"))[0].click();
                    }else{
                       window.open(res.url)
                    }
                    }else{
                      tool.tip(res.Message);
                    }
                }
            })
  • a标签download属性
  • 方案二 :补充方案 :利用form表单提交下载文件(ajax无法直接处理返回的文件类型),用于解决window.open方案被浏览器拦截的情况。
let $form = $("<form>") ;
$form.attr({method:"get",action:res.Message}).hide();
let queryStr = res.Message.split("?")[1];
let queryObj = qs.parse(queryStr) ;
$("body").append($form);
for(let p in queryObj){
    let $input =$("<input type='hidden'>") ;
    $input.attr({"name":p,value:queryObj[p]}).appendTo($form);
}
$form.submit().remove(); 

方案三 :前端利用download模块进行下载

支持场景
与上面的方案相比,这个模块提供的方案更加完善,而不是局限于某种方案,使用率很高。在源码中,我们可以看到在这个模块中针对各个浏览器和相应的属性是否支持进行了比较全面的兼容。其对应的下载文件方案包括了以下几种。

  • window.open(url)打开某个文件地址
  • iframe的框架中,设置src属性,通过iframe进行文件的下载,支持文件地址
  • 通过form标签,设置action的文件地址,然后通过form的提交来完成文件的下载(支持二进制)

方案小结:
对于常规的支持文件地址的下载,兼容性非常好,而对于传统的文件流性质的,通过form标签也可以进行简单的支持,可以说是非常好的方案了。当然如果你需要那么全面的方案,大多数情况用其中一个就可以了。

方案四 :h5新生方案下载

这个我觉得张鑫旭大佬介绍的蛮多的,应该上手足够了,就不多介绍了。除了a标签提供的download属性,多介绍了一种html:blob的方式。另外针对图片可以通过base64的方式。

传送门:h5新方式下载文件

个人建议:虽然新技术很好,但酌情使用,而且这里没有考虑任何兼容,也没有谈论到其他的一些文件类型,比如表格,pdf,大文件,视频音频的下载情况等。所以不是很建议把这个当做很常规的方案来考虑。

方案五 :file-saver

模块地址https://npm.taobao.org/package/file-saver,推荐使用,下载量以及稳定性足够好。支持新出的h5特性的加载方式,也就是方案五的部分。
github托管地址:https://github.com/eligrey/FileSaver.js

在模块的介绍中:详细说明了浏览器支持的情况,以及可以支持的下载范围,保存为的文件类型,与其我们去用基础知识踩雷,还是建议大家用成熟的模块方案去解决需求相关的问题。支持不了就退步用传统的方案解决,让后端提供直接的文件地址,要知道后端有更多的成熟的技术架包,对于前端来说还是萌新不确定的方案,后端早已经有了答案。

说明:我们之前的需求是希望下载一个表格文件,之前的方案是用后端生成文件地址,然后进行下载,其设置的返回response content type 为application/vnd.ms-excel (常规类型application/json)。后面发现有这个模块,基本使用还是体验蛮好的,此时的约定变成了后端根据查询的数据生成一个二进制的文件流,这样的好处是如果么有必要的时候可以减少在阿里云或者其他服务器暂存很多文件。

拓展思考下:在大家的公司里有没有遇到过类似的需求,按照我之前的经验是本来是想后端返回一个生成之后的文件地址,但后端的回复是由于采用了负载均衡,这个地址再去请求时不一定会请求到这个服务器,所以之前的前后端协调方案是放到了阿里云,然后通过设置权限和时效来保证文件的临时性,用户也可以在相似请求时不用重复请求数据库,重新生成文件,因为重复的数据内容会直接返回已经上传到阿里云的文件地址。

源码解析:
在其源码中,主要是针对返回的http的resonsetype做了要求,然后针对返回的地址进行处理,其中涉及到重要的代码:

//利用a标签下载
var a = document.createElement('a')
a.href = blob
//触发点击事件
node.dispatchEvent(new MouseEvent('click'))
// reader 进行解析
var reader = new FileReader()
var url = reader.result
//得到可解析的地址
_global.URL || _global.webkitURL,
URL.createObjectURL(blob)
//对 cors 跨域是否支持
  return xhr.status >= 200 && xhr.status <= 299

filereader的官方介绍:https://developer.mozilla.org/zh-CN/docs/Web/API/FileReader

总结

综上,无论是偏传统的方案,还是比较全面兼容的根据文件地址下载的方式,还是h5新出的webapi的方式都有比较好的认识,如果你对相关的知识点或者方案有进一步研究的兴趣,建议针对官方api的相关文章或者已经开源出的两个模块进行深度的优化和研究效率更佳。

觉得还不错,给个赞加关注吧,谢谢大家的支持。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,723评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,485评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,998评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,323评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,355评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,079评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,389评论 3 400
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,019评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,519评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,971评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,100评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,738评论 4 324
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,293评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,289评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,517评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,547评论 2 354
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,834评论 2 345

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,515评论 25 707
  • 用两张图告诉你,为什么你的 App 会卡顿? - Android - 掘金 Cover 有什么料? 从这篇文章中你...
    hw1212阅读 12,693评论 2 59
  • 青海湖 前两天的旅游我还在暗暗自喜,自己体质不错没有一点高原反应,没想第三天入住在藏区的青海湖畔的夜晚,我却头疼地...
    徜徉未来阅读 353评论 1 1
  • 努力机敏敢于放弃 对中下差生 具体抽象半抽象 半扶不扶 鱼的记忆职业事业师德 智慧能力苦难的意义美酒信仰 睡装睡醒...
    pretty1994阅读 83评论 0 0
  • 方世平—8月18日分析 加拿大7月份消费者物价指数(CPI)年率上升了3%,远高于2.5%的预期,也是自2011年...
    方世平阅读 210评论 0 0