认识Js中的二进制数据
Blob
在项目中涉及到要对html原生的audio组件进行样式复写,因此需要重新实现audio的一些功能,比如下载。实现一个下载大致的思路是服务端返回一段音频的二进制数据,客户端将其存放在Blob中,再通过URL.createObjectURL将其转换成blob url,最后动态创建a标签,添加download属性,模拟点击事件来实现下载。代码比较简单,我们重点来看看Blob是何方神圣。
const saveExcelFile = (blob, filename) => {
if (window.navigator.msSaveOrOpenBlob) {
// iE下使用msSaveBlob进行导出
navigator.msSaveBlob(blob, filename)
} else {
var href = window.URL.createObjectURL(blob)
var save_link = document.createElementNS(
'http://www.w3.org/1999/xhtml',
'a'
)
save_link.href = href
save_link.download = filename
// 解决火狐兼容问题
document.body.appendChild(save_link)
var event = document.createEvent('MouseEvents')
event.initMouseEvent(
'click',
true,
false,
window,
0,
0,
0,
0,
0,
false,
false,
false,
false,
0,
null
)
save_link.dispatchEvent(event)
document.body.removeChild(save_link)
// 释放blob url,可被GC回收
window.URL.revokeObjectURL(href)
}
}
根据MDN的介绍,我们可以知道Blob类型的对象是类似文件对象的二进制数据,它是immutable的,即数据不可变。而HTML5的File对象继承于Blob对象,并在其基础上做了些扩展,从而具备了在操作系统上操作文件的能力。我们可以利用Blob去做一些下载文件、分片上传等功能。熟悉ES6的小伙伴应该知道,ES6中有一个ArrayBuffer对象,也是用来存储二进制数据的,那它和Blob有什么区别呢?
ArrayBuffer
根据一些资料,ArrayBuffer设计的目的与WebGL项目有关,为了满足JS与显卡或声卡等操作系统原生接口大量的、实时的数据交换。传统的文本格式是传递一个32位的整数,这导致JS与原生接口需要频繁的转换数据格式,效率较低,因此设计了ArrayBuffer用于存储、操作二进制数据。
ArrayBuffer并不是真正的Array,而是个类数组对象。我们通过new ArrayBuffer(length)创建的ArrayBuffer的实例,仅仅代表开辟了一段连续的内存空间,length代表内存所占的字节大小。若需要对内存中的字节进行操作,则需要创建“视图”。视图分为两种:TypedArray和DataView,用于以指定的格式来读写二进制的数据。它们的区别在于:
- TypedArray以指定的格式读写内存,例如:const v1 = new Int32Array(buffer)就是以32位有符号整型来创建视图,此时通过v1[0]去读或是去写都是以该格式进行的
- 若是想以不同数据格式去读取内存的话,需要使用DataView。当我们执行const dv = new DataView(buffer)后,可以通过类似dv.getUint8(0)这样的方式,以8位无符号整型读取第一个字节;或是以dv.setInt32(1, 25)这种方式,在第二个字节写入值为25的32位有符号整型数据
注意:对于同一段内存创建的视图都是共享该内存的,在一个视图上进行的操作会影响另一个视图的读写。具体的可参考阮老师的教程
区别
- ArrayBuffer可以对字节进行读写,而Blob是immutable的
- ArrayBuffer存储在内存当中,Blob可以存储在磁盘或者内存中。例如文件,我们平时是存在磁盘中的。而像我们上面下载的例子中,我们的blob是在内存中的,因此在createObjectURL之后需要手动调用revokeObjectURL解除对内存的引用,使得blob可以被GC回收,释放内存。
- ArrayBuffer可以通过“视图”来进行读写,而Blob可以通过FileReader去读,但是不能写
- Blob和ArrayBuffer可以互相转换。Blob转ArrayBuffer可以通过:
const reader = new FileReader()
reader.onload = function() {
console.log(reader.result)
}
reader.readAsArrayBuffer(blob)
ArrayBuffer转Blob可以通过:
const blob = new Blob([ArrayBuffer])
因此,当我们需要对字节进行操作的时候,我们应该选用ArrayBuffer,否则,我们用Blob会更加容易。
认识Js中的二进制数据的更多相关文章
- Js中各类型数据到bool的转换
在返回Json字符串给前台时遇到的问题,返回的bool数据总是为TRUE 特意查了一下,发现了Js中各类数据转换到bool型是的结果. 希望能给遇到同样问题的人一点帮助. 数据类型 转换为bool ...
- 解决js中post提交数据并且跳转到指定页面的问题总结
今天在开发中过程中遇到了这个问题,js中利用JQuery中的 $.post("url", id, function(){}); 这个方法是数据提交正常,但是后台处理完成之后跳转无法 ...
- js中如何将数据获得2位小数以及对数据进行千分位划分
js中toFixed(n) 方法可把 数字四舍五入为指定小数位数n的数字,注意:这个方法只能对数据类型为Number的数据起作用,包括float,int等.例如: 123.12345.toFixe ...
- JS中浏览器的数据存储机制
一.JS中的三种数据存储方式 cookie.sessionStorage.localStorage 二.cookie 1.cookie的定义: cookie是存储在浏览器上的一小段数据,用来记录某些当 ...
- JS中关于引用类型数据及函数的参数传递
(JavaScript 中,函数的参数传递方式都是按值传递,没有按引用传递的参数) 一.数据类型 在 javascript 中数据类型可以分为两类: 基本类型值 primitive type,比如Un ...
- JSBinding+Bridge:逻辑代码中操作二进制数据
以这2个函数为例 class File { public static byte[] ReadAllBytes(string path); public static void WriteAllByt ...
- js中Math()函数&&数据类型转换
Math()函数: x的y次方:Math.pow(x,y) 取小数点后两位:num.toFixed(2) 数据类型转换: 字符串转换为数字:parseInt(num)转换为整数:parseFloat( ...
- python struct.pack() 二进制文件,文件中打包二进制数据的存储与解析
学习Python的过程中,遇到一个问题,在<Python学习手册>(也就是<learning python>)中,元组.文件及其他章节里,关于处理二进制文件里,有这么一段代码的 ...
- js 中 前端过滤数据到后端的方法
第一种方法: <!DOCTYPE html><html lang="en"><head> <meta charset="UTF- ...
随机推荐
- 单调栈&单调队列入门
单调队列是什么呢?可以直接从问题开始来展开. Poj 2823 给定一个数列,从左至右输出每个长度为m的数列段内的最小数和最大数. 数列长度:\(N <=10^6 ,m<=N\) 解法① ...
- Windows环境下,本地Oracle创建dblink连接远程mysql
前言 我的情况是,本地安装了oracle(安装完成后带有SQL Developer,不需要再安装instantclient),创建dblink去连接远程的mysql.有些朋友可能是 本地使用PL\SQ ...
- CSS grayscale滤镜+SVG使图片变黑白实例页面
http:/CSS 地址:/www.runoob.com/cssref/css3-pr-filter.html CSS代码: .gray { -webkit-filter: grayscale(%); ...
- 漏测BUG LIst
5. 接口设计问题 - 主从存在延时,当两个接口需要一个主库,一个从库的时候,可能会出问题,时时性 4. 开发的接口文档也得进行简单的测试,根据产品文档/业务测试接口(针对问题2) 3. 需要上的课 ...
- python中的单向链表实现
引子 数据结构指的是是数据的组织的方式.从单个数据到一维结构(线性表),二维结构(树),三维结构(图),都是组织数据的不同方式. 为什么需要链表? 顺序表的构建需要预先知道数据大小来申请连续的存储空间 ...
- vue font-icon 图标
1.vue 游览器左上角小图标 把.ico文件放在根目录下的static文件夹下,然后link标签引入 <link rel="shortcut icon" href=&quo ...
- python 用正则处理日志实例
前提: 了解正则基本语法 import re with open('top10_xiaozhuang_net.log','r') as f1: #读取日志文件 subject=f1.rea ...
- git常用方法整理
Git是什么? Git是目前世界上最先进的分布式版本控制系统(没有之一). Git有什么特点?简单来说就是:高端大气上档次! 初始化本地仓库 mkdir xxx cd xxx git init 创建本 ...
- c# 通过MailHelper发送QQ邮件
发送的方法 appsetting内容 第一个是发送邮件qq账号,第二个是QQ邮箱的POP3/SMTP服务码(下面会说怎么获取),第三个是服务器,第四个是端口 获取QQ邮箱的POP3/SMTP服务码 1 ...
- TCP协议学习总结(上)
在计算机领域,数据的本质无非0和1,创造0和1的固然伟大,但真正百花齐放的还是基于0和1之上的各种层次之间的组合(数据结构)所带给我们人类各种各样的可能性.例如TCP协议,我们的生活无不无时无刻的站在 ...