ES6躬行记(13)——类型化数组

本文涉及的产品
可视分析地图(DataV-Atlas),3 个项目,100M 存储空间
简介:   类型化数组(Typed Array)是一种处理二进制数据的特殊数组,它可像C语言那样直接操纵字节,不过得先用ArrayBuffer对象创建数组缓冲区(Array Buffer),再映射到指定格式的视图(view)之后,才能读写其中的数据。总共有两类视图,分别是特定类型的TypedArray和通用类型的DataView。在ES6引入类型化数组之后,大大提升了JavaScript数学运算的性能。

  类型化数组(Typed Array)是一种处理二进制数据的特殊数组,它可像C语言那样直接操纵字节,不过得先用ArrayBuffer对象创建数组缓冲区(Array Buffer),再映射到指定格式的视图(view)之后,才能读写其中的数据。总共有两类视图,分别是特定类型的TypedArray和通用类型的DataView。在ES6引入类型化数组之后,大大提升了JavaScript数学运算的性能。


一、ArrayBuffer


  虽然ArrayBuffer对象可以开辟一片固定大小的内存区域(即数组缓冲区),但它不能直接读写所存储的数据,需要借助视图才行。通过构造函数ArrayBuffer()可以分配指定字节数量的缓冲区,如下代码所示,分配了一段8个字节的内存区域,每个字节的默认值都为0。有一点要注意,缓冲区的容量在指定后,就不可再修改。

var buffer = new ArrayBuffer(8);

  利用ArrayBuffer对象的只读属性byteLength,可以获取缓冲区的字节长度。而通过它的slice()方法可创建一个指定范围的缓冲区副本。还有一个静态方法ArrayBuffer.isView(),用来检测是否是一个视图实例,具体如下所示。

buffer.byteLength;             //8
buffer.slice(2, 4);
ArrayBuffer.isView(buffer);    //false


二、TypedArray


1)数据类型

  视图的行为类似于数组,但并不是真正的数组。视图可根据指定的数值类型解读缓冲区中的二进制数据,而TypedArray包含9种特定类型的视图(即类型化数组),如表8所示(引用自ES6规范的22.2小节)。


表8  九种特定类型的视图


视图名称 元素大小(字节) 描述 等价的C语言类型
Int8Array 1 8位二进制补码有符号整数 signed char
Uint8Array 1 8位无符号整数 unsigned char
Uint8ClampedArray 1 8位无符号整数(对溢出做特殊处理) unsigned char
Int16Array 2 16位二进制补码有符号整数 short
Uint16Array 2 16位无符号整数 unsigned short
Int32Array 4 32位二进制补码有符号整数 int
Uint32Array 4 32位无符号整数 unsigned int
Float32Array 4 32位IEEE浮点数 float
Float64Array 8 64位IEEE浮点数 double


  表中的9种视图都继承自TypedArray对象,而每种视图都只能处理一种数值类型的数据(例如Float32Array只能处理float类型的数值),并且视图中的元素大小(即所占的字节数)也与数值类型有关,例如Float32Array的1个元素包含4个字节。由于各个视图的元素都会有规定的数值范围,因此超出该范围就会溢出。其中对Uint8ClampedArray的溢出处理较为特殊,它的数值范围在0到255之间,如果缓冲区所存的值超出该范围,那么就会替换这个值,例如小于0的数值被转换成0,而大于255的数值则被转换成255,如下所示。

var view = new Uint8ClampedArray(2);
view[0] = -1;
view[1] = 256;
console.log(view);        //Uint8ClampedArray(2) [0, 255]

2)创建

  每种特定类型的视图都有一个构造函数,每个构造函数都有4种参数的组合,即共有4种方式创建类型化数组,具体如下所列,每一种创建方式的后面都会给出相应的示例代码。

  (1)3个参数,第一个是数组缓冲区,第二个是可选的字节偏移量,第三个是可选的需要包含的元素个数。注意,偏移量需要是元素大小的倍数。

var buffer = new ArrayBuffer(16),
  view1 = new Int16Array(buffer),            //Int16Array(8) [0, 0, 0, 0, 0, 0, 0, 0]
  view2 = new Int16Array(buffer, 4, 2);        //Int16Array(2) [0, 0]

  利用这组参数,可以让一个缓冲区关联多个视图,如下代码所示。

var view1 = new Int8Array(buffer, 0, 4),
  view2 = new Int16Array(buffer, 4, 4),
  view3 = new Int32Array(buffer, 12, 1);
view1.buffer === view2.buffer;            //true
view2.buffer === view3.buffer;            //true

  三个视图分别占据了前4个字节、中间8个字节以及后4个字节的缓冲区。视图的buffer属性指向了它所处的缓冲区,两组全等比较的结果都为真,由此可知,它们使用了同一个缓冲区。

  (2)1个数值,表示类型化数组的元素个数,将该参数乘以每个元素的大小即可得到缓冲区的容量,前面描述Uint8ClampedArray特性的示例就用到了这种创建方式。注意,此时构造函数会自动创建合适容量的缓冲区。

var view = new Int16Array(7);             //Int16Array(7) [0, 0, 0, 0, 0, 0, 0]

  (3)1个类型化数组,它的元素会被复制到新的类型化数组中,虽然元素个数保持不变,但使用了不同的缓冲区。

var view1 = new Int16Array(6),
  view2 = new Int8Array(view1);           //Int8Array(6) [0, 0, 0, 0, 0, 0]

  (4)1个对象,只要不是TypedArray或ArrayBuffer就行,例如数组、类数组等对象。

var view = new Int16Array([1, 2, 3]);     //Int16Array(3) [1, 2, 3]

3)与常规数组的异同

  类型化数组与常规数组有许多相似点,下面仅列出其中的三点:

(1)都可以通过数字类型的索引来访问某个位置的元素。

(2)通过length属性可获取包含的元素个数。

(3)都包含slice()、of()、from()、copyWithin()等数组方法。

  虽然两者之间的共性不少,但是类型化数组的特点又很鲜明,例如:

  (1)每种类型化数组都包含一个BYTES_PER_ELEMENT属性,能获取每个元素所占的字节(即元素大小),如下代码所示。

Int8Array.BYTES_PER_ELEMENT;         //1
Int16Array.BYTES_PER_ELEMENT;        //2
Int32Array.BYTES_PER_ELEMENT;        //4

  (2)由于类型化数组无法维护自己的长度,因此将length属性定义为只读,并且缺少pop()、push()、shift()等会更改数组长度的方法。

  (3)包含独有的属性和方法,如表9所列。


表9  类型化数组的属性和方法

属性或方法 功能描述
buffer 只读属性,读取视图所在的数组缓冲区
byteOffset 只读属性,读取字节偏移量
byteLength 只读属性,读取视图所占的字节长度
set() 从另一个常规数组或类型化数组中提取元素,再复制给当前类型化数组
subarray() 从当前类型化数组中提取元素,再由这些元素组成一个新的类型化数组

  byteOffset属性等于构造函数的第二个参数,而byteLength属性等于构造函数的第三个参数与元素大小相乘的积,如下所示。

var buffer = new ArrayBuffer(8),
  view = new Int16Array(buffer, 2, 3);
view.buffer;            //ArrayBuffer(8) {}
view.byteOffset;        //2
view.byteLength;        //6

  set()方法可接收2个参数,第一个是被提取的常规数组或类型化数组,第二个是可选的参数,表示当前类型化数组的偏移量,即从这个位置开始覆盖它的元素。subarray()方法也能接收2个参数,但都是可选的索引参数,第一个是开始提取的位置,第二个是结束提取的位置,注意,该位置上的元素不会被提取。关于两个方法的使用,可参考下面的代码。

view.set([8], 1);
console.log(view);        //Int16Array(3) [0, 8, 0]
view.subarray(1, 2);      //Int16Array [8]


三、DataView


  DataView只有一个身份(即视图),而之前的类型化数组有双重身份,既是视图,也是类数组。如果要使用DataView视图,那么需要先创建数组缓冲区,类似于类型化数组的第一种创建方式,只不过它的构造函数中的第三个可选的参数换成需要包含的字节长度,如下代码所示。根据全等比较的结果可知,两个视图处在同一个缓冲区中。

var buffer = new ArrayBuffer(16),
  view1 = new DataView(buffer),
  view2 = new DataView(buffer, 4, 6);
view1.buffer === view2.buffer;            //true

1)写入和读取

  想要通过DataView视图读写缓冲区的数据,需要先为其指定数据类型,而它支持8种数据类型(除了Uint8Clamped)。DataView提供了8对原型方法,每对方法分别以“set”和“get”作为名称前缀,前者用于写入,后者用于读取,在前缀的后面会跟着数据类型,例如setInt16()和getInt16()。

  写入方法的第一个参数是字节偏移量,第二个参数是要定义的数值。而读取方法的第一个参数也是字节偏移量,如下代码所示,两张视图被指定了不同的数据类型。

view1.setInt8(2, 8);
view1.getInt8(0);        //0
view1.getInt8(2);        //8
view2.setInt16(0, 16);
view2.getInt16(0);       //16
view2.getInt16(2);       //0

2)小端序

  除了Int8和Uint8类型,其它类型的写入和读取方法都还包含一个可选的布尔参数:littleEndian,表示是否启用小端序,默认为true。在了解小端序之前,需要先了解一下端序。端序又称字节序(Endianness),表示多字节中的字节排列方式。小端序是指字节的最低有效位在最高有效位之前(大端序正好与之相反),例如数字10,如果用16位二进制表示,那么它就变为0000 0000 0000 1010,换算成16进制就是000A,用小端序存储的话,该值会被表示成0A00。虽然大端序更符合人类的阅读习惯,但英特尔处理器和多数浏览器采用的都是小端序。引入该参数后,能更灵活的处理不同存储方式的数据。


相关实践学习
DataV Board用户界面概览
本实验带领用户熟悉DataV Board这款可视化产品的用户界面
阿里云实时数仓实战 - 项目介绍及架构设计
课程简介 1)学习搭建一个数据仓库的过程,理解数据在整个数仓架构的从采集、存储、计算、输出、展示的整个业务流程。 2)整个数仓体系完全搭建在阿里云架构上,理解并学会运用各个服务组件,了解各个组件之间如何配合联动。 3 )前置知识要求   课程大纲 第一章 了解数据仓库概念 初步了解数据仓库是干什么的 第二章 按照企业开发的标准去搭建一个数据仓库 数据仓库的需求是什么 架构 怎么选型怎么购买服务器 第三章 数据生成模块 用户形成数据的一个准备 按照企业的标准,准备了十一张用户行为表 方便使用 第四章 采集模块的搭建 购买阿里云服务器 安装 JDK 安装 Flume 第五章 用户行为数据仓库 严格按照企业的标准开发 第六章 搭建业务数仓理论基础和对表的分类同步 第七章 业务数仓的搭建  业务行为数仓效果图  
相关文章
|
4月前
|
开发者 Python 容器
python函数基础以及函数参数简解
python函数基础以及函数参数简解
|
存储 NoSQL 测试技术
SystemVerilog学习-05-数组
SystemVerilog学习-05-数组
961 0
SystemVerilog学习-05-数组
|
前端开发
前端学习笔记202306学习笔记第三十八天-Es6-对象得解构赋值2
前端学习笔记202306学习笔记第三十八天-Es6-对象得解构赋值2
62 0
|
前端开发
前端学习笔记202306学习笔记第三十八天-Es6-字符串的解构赋值1
前端学习笔记202306学习笔记第三十八天-Es6-字符串的解构赋值1
59 0
前端学习笔记202306学习笔记第三十八天-Es6-字符串的解构赋值1
|
前端开发
前端学习笔记202306学习笔记第三十八天-Es6-数组得结构赋值
前端学习笔记202306学习笔记第三十八天-Es6-数组得结构赋值3
56 0
|
前端开发
前端学习笔记202306学习笔记第三十八天-Es6-数组得结构赋值3
前端学习笔记202306学习笔记第三十八天-Es6-数组得结构赋值3
48 0
|
前端开发
前端学习笔记202306学习笔记第三十八天-Es6-对象得解构赋值3
前端学习笔记202306学习笔记第三十八天-Es6-对象得解构赋值3
49 0
|
前端开发
前端学习笔记202306学习笔记第三十八天-Es6-函数形参的结构赋值1
前端学习笔记202306学习笔记第三十八天-Es6-函数形参的结构赋值1
54 0
|
前端开发
前端学习笔记202306学习笔记第三十八天-Es6-数组得结构赋值1
前端学习笔记202306学习笔记第三十八天-Es6-数组得结构赋值1
55 0
|
前端开发
前端学习笔记202306学习笔记第三十八天-Es6-数组得结构赋值2
前端学习笔记202306学习笔记第三十八天-Es6-数组得结构赋值2
53 0