前端文件的下载

file对象,blob对象,filreader对象,fileList和dataTransfer。
生成下载链接,a.download属性进行下载 。

<a>的download属性

一般情况下我们都是将文件上传到后端存储,然后再根据需要通过后端进行文件下载。这种情况需要后端来告知浏览器文件需要下载而不是打开,这样前后端都需要操作是令人不悦的。
终于,H5为<a>提供了一个新属性download,一旦a标签具有down属性,点击该标签后就会自动下载对应href下的资源文件。比如:

<a href="http://www.w3school.com.cn/i/w3school_logo_white.gif" download="logo.gif">点我下载</a>

当然,直接在你的html中使用上面的代码还是无法下载,因为存在跨域资源限制
上面实例中的a标签的href的链接指向的文件仍然是后端数据,现在我们可以使用js提供的一些方法来创建特别的URL提供文件下载:
目前主要有两种URL,

  • BlobURL
  • DataURl

Blob对象

Blob全称binary large object(二进制大对象),顾名思义,Blob对象就是用来存储二进制信息的“容器”。

#使用构造函数将非Blob对象转换为Blob对象

Blob(blobParts[, options])
Blob() 构造函数返回一个新的 Blob 对象。 Blob的内容由参数数组中给出的值的串联组成。
很简单,我们把要转化的数据放到数组中作为参数即可:

var str = "This is a userful message!"
var blob = new Blob([str]);//将字符串转化为Blob对象
#将Blob对象转化为BlobURL

我们上面提到的a标签能打开BlobURL中的数据,这样我们就可以下载我们处理过的数据,步骤即:
(即将数据转为Blob,再将Blob生成BlobURL,使用a标签下载。)

var url = URL.createObjectURL(blob);

下面我们写一个方法来根据给定的文件内容和download动态下载文件。

#动态下载文件
function downloadFile(content, filename) {
    // 创建a链接
    var eleLink = document.createElement('a');
    eleLink.download = filename;
    eleLink.style.display = 'none';
    // 字符内容转变成BlobURL
    var blob = new Blob([content]);
    eleLink.href = URL.createObjectURL(blob);
    // 触发点击(开始下载)
    document.body.appendChild(eleLink);
    eleLink.click();
    // 移除标签
    setTimeout(function(e){
        document.body.removeChild(eleLink);
    },1000);
};

测试:

<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>Test</title> 
<script src="https://cdn.staticfile.org/jquery/1.10.2/jquery.min.js"></script>
</head>
<body>
    <p onclick="run()">点我下载</p>
    <script>
        function downloadFile(content, filename) {
            // 创建a链接
            var eleLink = document.createElement('a');
            eleLink.download = filename;
            eleLink.style.display = 'none';
            // 字符内容转变成BlobURL
            var blob = new Blob([content]);
            eleLink.href = URL.createObjectURL(blob);
            // 触发点击(开始下载)
            document.body.appendChild(eleLink);
            eleLink.click();
            // 移除标签
            setTimeout(function(e){
                document.body.removeChild(eleLink);
            },1000);
        };
        function run(){
            var content = "This is a userful message!";
            var filename = "message.txt";
            downloadFile(content,filename);
        }
    </script>
</body>
</html>

理论上可以使用Blob对象下载任意类型的文件,毕竟都能转化为二进制文件,但目前主要用来存储文本文件。况且对于非文本文件或体积很大的文件,参数的传入是个问题。所以我们有必要学习File对象。

File对象

File对象继承了Blob并进行了扩展,使其能够支持使用系统上的文件,如txt等常见文件。
比如我们有一张png,我们如何进行下载呢?当然可以使用Blob对象,但是参数又该如何写呢,这是很麻烦的。
这个时候我们就可以使用File对象获取png,而File继承自Blob,所以也能转换为BlobURL,但是File还有一个好处就是File对象可以从页面中获取文件(input标签),拖放操作生成的 DataTransfer对象。
我们从下面的例子中观察:

<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>File对象和FileList对象</title>
</head>
<body>
    <form method="post" type="multipart/form-data" name="fileform">
        <label>导入多个文件:
            <input type="file" multiple name="myfile">
        </label>
        <label>导入单个文件:
            <input type="file" name="myfile2">
        </label>
        <input type="submit" value="提交">
    </form>
    <script>
        $(function () { 
            var myform = document.forms["fileform"];
            var inputs = myform.querySelectorAll("input");
            var files;//fileList对象
            var file2;//file对象
            myform.onsubmit = function(e){
                files = inputs[0].files;//fileList对象
                file2 = inputs[1].files[0];//file对象
                console.log(files);
                console.log(file2);
                e.preventDefault();//阻止默认处理,否则页面可能会被刷新
            }
        });
    </script>
</body>
</html>

总结一下:
input的files属性包含了上传的文件,当input使用multiple属性时可上传多个文件。其中,每个本件就是一个File对象。
现在我们就能使用URL.createObjectURL(file);来创建BlobURL链接了。

DataURL

上面都只说到BlobURL,其实还有一种链接,即DataURL,但这种链接是FileReader对象生成的。
一般情况下BlobURL已经足够,但是经过多次尝试,在手机端貌似无法打开BlobURL,也就无法实现下载,而DataURL却可以打开。(但是DataURL有时候也会失败!且在PC端也容易失败)
FileReader的相关方法可以对Blob对象和File对象进行读取:


其中,
FileReader.readAsDataURL()是我们需要的方法,通过该方法,FileReader将读取Blob或File对象并使得result属性的值包含data:URL格式的链接,即DataURL。

简单的总结

  • a标签的download属性可以告知浏览器直接下载href指定的文件。
  • href除了正常正常的URL链接,还支持BlobURL和DataURL。(是具有生命周期的)
  • 我们保存文件数据可以使用Blob对象或File对象。
  • 使用URL.createObjectURL(blob/file);创建BlobURL.
  • 使用FIleReader.readAsDataURL(blob/file)创建DataURL.
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,444评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,421评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,363评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,460评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,502评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,511评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,280评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,736评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,014评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,190评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,848评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,531评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,159评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,411评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,067评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,078评论 2 352

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,646评论 18 139
  • simplify the life HTML5 File API — 让前端操作文件变的可能 前言 在 HTML5...
    我是强强阅读 3,293评论 0 1
  • 用两张图告诉你,为什么你的 App 会卡顿? - Android - 掘金 Cover 有什么料? 从这篇文章中你...
    hw1212阅读 12,709评论 2 59
  • 闲言: 上篇我们学习了一些下载的js前端方式,这篇来讲讲上传篇。这也是在项目需求中很常见的O(∩_∩)O。 dem...
    迷缘火叶阅读 546评论 0 1
  • 2017年12月7号 相博超妈妈 今天下午两点接到老师的电话,说相博超头疼让我去接他,我就赶紧过去了,领他去卫生...
    一年级四班相博超妈妈阅读 102评论 0 1