博客地址:https://ainyi.com/88html
对于 Blob,前端开发中可能比较少遇到;数据库中可以使用 Blob 概念,例如 Mysql 存储二进制数据的类型就是 Blob,也就是说图片可存储于数据库中,以二进制格式存储前端
Blob 对象表示一个不可变、原始数据的类文件对象。File 接口基于Blob,继承了 blob 的功能并将其扩展使其支持用户系统上的文件ios
Blob 是二进制数据对象,是类文件对象的二进制数据ajax
我在以前有篇博客说到 Blob:利用 Blob 处理 Node 层返回的二进制文件流字符串并下载文件
这里我利用 Blob 实现文件拆分再合并下载的方法,算是第一次使用sql
咱们最多见的应该是 Blob URL 技术,文件上传的预览、视频播放的 src,均是采用这种技术实现数据库
Blob URL 就是以 blob: 开头的一段地址,指向的是一个二进制数据
使用 URL.createObjectURL(blob) 方法生成,参数为 Blob 对象axios
这个 Blob URL 是能够直接访问的;须要注意的是这个 URL 的生效时间,等同于网页的存在时间,一旦网页刷新或关闭,这个 Blob URL 就失效后端
Blob(blobParts[, options])
返回一个新建立的 Blob 对象,其内容由参数中给定的数组串联组成数组
参数说明:
blobParts:数组类型,数组中的每一项链接起来构成 Blob 对象的数据,数组中的每项元素能够是ArrayBuffer, ArrayBufferView, Blob, DOMStringapp
options:可选参数;字典格式类型,能够指定以下两个属性:
介绍三种使用场景
// 获取文件二进制流 content const content = await downloadContract(params) // 再利用 Buffer 转为对象 const buf = Buffer.from(content, 'binary') // 生成 Blob 对象,type 类型设置为 pdf 的 MIME 类型 const blob = new Blob([buf], {type: 'application/pdf'}); // 获取 Blob URL,可赋值到 a 标签 href 属性进行下载 const url = URL.createObjectURL(blob)
经过 Blob 生成文件、利用 Blob URL 获取下载连接,这样就实现后端返回二进制格式的文件进行合并再下载
较为简单,获取文件对象后,再经过 createObjectURL 方法获得 Blob URL
最后直接赋值到 img 标签的 src 属性便可
<input id="upload" type="file" /> <img id="preview" src="" alt="预览"/>
const upload = document.querySelector('#upload') const preview = document.querySelector('#preview') upload.onchange = function() { const file = upload.files[0] // File 对象 const src = URL.createObjectURL(file) preview.src = src }
视频地址,不一样于上面的 input,能够直接拿到 File 对象
只有一个视频地址怎么能将这个 URL 变成咱们想要的 Blob URL 形式呢
从 URL.createObjectURL(blob) 方法来看,首先要拿到存储这个视频原始数据的 Blob 对象
平时咱们请求接口可使用 axios / ajax / xhr 或 fetch,请求一个服务端地址能够返回咱们相应的数据,那若是咱们去请求一个图片或视频地址会返回什么?应当是返回图片和视频的数据,这种状况只要设置正确responseType才能拿到咱们想要的格式数据
// responseType 参数以下: // text 字符串;blob Blob对象;arraybuffer ArrayBuffer 对象 function ajax(url, cb) { const xhr = new XMLHttpRequest() xhr.open('get', url) xhr.responseType = 'blob' xhr.onload = function() { cb(xhr.response) } xhr.send() }
上面请求返回一个 Blob 对象,接下来只要而后经过 createObjectURL 生成 Blob URL 赋值给视频的 src 属性就能够了
ajax('video.mp4', function(res){ const src = URL.createObjectURL(res) video.src = src })
最近看到一篇文章:大规格文件的上传优化
里面讲的是利用 Blob 实现文件分片上传,对于大文件上传有很好的效果
其核心思想是文件分片,使用 File.slice() 方法进行文件分片;File 对象是继承 Blob 对象的,所以 File 对象也有 slice 方法
Blob.slice([start[, end[, contentType]]])
start 可选
这个参数表明 Blob 里的下标,表示第一个会被会被拷贝进新的 Blob 的字节的起始位置。若是你传入的是一个负数,那么这个偏移量将会从数据的末尾从后到前开始计算
举例来讲: -10 将会是 Blob 的倒数第十个字节。它的默认值是0, 若是你传入的start的长度大于源 Blob 的长度,那么返回的将会是一个长度为0而且不包含任何数据的一个 Blob 对象
end 可选
这个参数表明的是 Blob 的一个下标,这个下标-1的对应的字节将会是被拷贝进新的Blob 的最后一个字节。若是你传入了一个负数,那么这个偏移量将会从数据的末尾从后到前开始计算
举例来讲: -10 将会是 Blob 的倒数第十个字节。它的默认值就是它的原始长度(size)
contentType 可选
给新的 Blob 赋予一个新的文档类型。这将会把它的 type 属性设为被传入的值。它的默认值是一个空的字符串
文件分片方法
定义每个分片文件的大小变量为 chunkSize,经过文件大小 FileSize 和分片大小 chunkSize 获得分片数量 chunks,使用 for 循环和 file.slice() 方法对文件进行分片,序号为 0 - n,和已上传的切片列表作比对,获得全部未上传的分片,push 到请求列表 requestList
上传进度
监听原生 Javascript 的 XMLHttpRequest 的 progress 事件,这个事件会返回文件已上传的大小和总大小,可实现上传进度的变化
博客地址:https://ainyi.com/88