JavaScript 类型化数组

JavaScript 类型化数组(typed array)是一种类似数组的对象,并提供了一种用于在内存缓冲区中访问原始二进制数据的机制。

Array 存储的对象能动态增多和减少,并且可以存储任何 JavaScript 值。JavaScript 引擎会做一些内部优化,以便对数组的操作可以很快。然而,随着 Web 应用程序变得越来越强大,尤其一些新增加的功能例如:音频视频编辑、访问 WebSockets 的原始数据等,很明显有些时候如果使用 JavaScript 代码可以快速方便地通过类型化数组来操作原始的二进制数据将会非常有帮助。JavaScript 类型化数组中的每一个元素都是原始二进制值,而二进制值采用多种支持的格式之一(从 8 位整数到 64 位浮点数)。

但是,不要把类型化数组与普通数组混淆,因为在类型数组上调用 Array.isArray() 会返回 false。此外,并不是所有可用于普通数组的方法都能被类型化数组所支持(如 push 和 pop)。

缓冲和视图:类型数组架构

为了达到最大的灵活性和效率,JavaScript 类型化数组将实现拆分为缓冲视图两部分。缓冲(由 ArrayBuffer 对象实现)描述的是一个数据分块。缓冲没有格式可言,并且不提供访问其内容的机制。为了访问在缓冲对象中包含的内存,你需要使用视图。视图提供了上下文——即数据类型、起始偏移量和元素数——将数据转换为实际有类型的数组。

ArrayBuffer 中的类型化数组

ArrayBuffer

ArrayBuffer 是一种数据类型,用来表示一个通用的、固定长度的二进制数据缓冲区。你不能直接操作 ArrayBuffer 中的内容;你需要创建一个类型化数组的视图或一个描述缓冲数据格式的 DataView,使用它们来读写缓冲区中的内容。

类型化数组视图

类型化数组视图具有自描述性的名字和所有常用的数值类型像 Int8Uint32Float64 等等。有一种特殊类型的数组 Uint8ClampedArray。它仅操作 0 到 255 之间的数值。例如,这对于 Canvas 数据处理非常有用。

类型 值范围 字节数 描述 对应的 Web IDL 类型 等效的 C 类型
Int8Array -128~127 1 8 位有符号整数(补码) byte int8_t
Uint8Array 0~255 1 8 位无符号整数 octet uint8_t
Uint8ClampedArray 0~255 1 8 位无符号整数(值会被裁剪) octet uint8_t
Int16Array -32768~32767 2 16 位有符号整数(补码) short int16_t
Uint16Array 0~65535 2 16 位无符号整数 unsigned short uint16_t
Int32Array -2147483648~2147483647 4 32 位有符号整数(补码) long int32_t
Uint32Array 0~4294967295 4 32 位无符号整数 unsigned long uint32_t
Float32Array -3.4E38~3.4E38 以及 1.2E-38(最小正数) 4 32 位 IEEE 浮点数(7 位有效数字,例如 1.123456 unrestricted float float
Float64Array -1.8E308~1.8E308 以及 5E-324(最小正数) 8 64 位 IEEE 浮点数(16 位有效数字,例如 1.123...15 unrestricted double double
BigInt64Array -263~263 - 1 8 64 位有符号整数(补码) bigint int64_t (signed long long)
BigUint64Array 0~264 - 1 8 64 位无符号整数 bigint uint64_t (unsigned long long)

DataView

DataView 是一种底层接口,它提供有可以操作缓冲区中任意数据的访问器(getter/setter)API。这对操作不同类型数据的场景很有帮助,例如:类型化数组视图都是运行在本地字节序模式(参考字节序),可以通过使用 DataView 来控制字节序。默认是大端字节序(Big-endian),但可以调用 getter/setter 方法改为小端字节序(Little-endian)。

使用类型化数组的 Web API

下面是一些使用类型化数组的 API 示例,其并不完整,我们会在未来添加更多示例。

FileReader.prototype.readAsArrayBuffer()

FileReader.prototype.readAsArrayBuffer() 读取对应的 BlobFile 的内容

XMLHttpRequest.prototype.send()

XMLHttpRequest 实例的 send() 方法现在支持使用类型化数组和 ArrayBuffer 对象作为参数。

ImageData.data

是一个 Uint8ClampedArray 对象,用来描述包含按照 RGBA 序列的颜色数据的一维数组,其值的范围在 0255 之间。

示例

使用视图和缓冲

首先,我们创建一个 16 字节固定长度的缓冲:

const buffer = new ArrayBuffer(16);

现在我们有了一段初始化为 0 的内存,目前还做不了什么太多操作。让我们确认一下数据的字节长度:

if (buffer.byteLength === 16) {
  console.log("Yes, it's 16 bytes.");
} else {
  console.log("Oh no, it's the wrong size!");
}

在实际开始操作这个缓冲之前,需要创建一个视图。我们将创建一个视图,此视图将把缓冲内的数据格式化为一个 32 位有符号整数数组:

const int32View = new Int32Array(buffer);

现在我们可以像普通数组一样访问该数组中的元素:

for (let i = 0; i < int32View.length; i++) {
  int32View[i] = i * 2;
}

该代码会将数组以 0246 填充(一共 4 个 4 字节元素,所以总长度为 16 字节)。

同一数据的多个视图

更有意思的是,你可以在同一数据上创建多个视图。例如:基于上文的代码,我们可以添加如下代码处理:

const int16View = new Int16Array(buffer);

for (let i = 0; i < int16View.length; i++) {
  console.log(`Entry ${i}: ${int16View[i]}`);
}

这里我们创建了一个 16 位整数视图,该视图共享上文的 32 位整数视图的缓冲,然后以 16 位整数打印出缓冲里的数据,这次我们会得到 00204060 这样的输出。

那么,这样呢?

int16View[0] = 32;
console.log(`Entry 0 in the 32-bit array is now ${int32View[0]}`);

这次的输出是 "Entry 0 in the 32-bit array is now 32"。也就是,这 2 个数组都是同一数据的以不同格式展示出来的视图。你可以使用任何一种视图类型

使用复杂的数据结构

通过将缓冲与不同类型视图组合,以及修改内存访问的偏移量,你可以操作包含更多更复杂数据结构的数据。例如,你可以操作诸如 WebGL 或数据文件这些复杂的数据结构。

请看如下的 C 语言结构体:

struct someStruct {
  unsigned long id;
  char username[16];
  float amountDue;
};

你可以采用如下代码访问一个包含此类结构体的缓冲:

const buffer = new ArrayBuffer(24);

// ... read the data into the buffer ...

const idView = new Uint32Array(buffer, 0, 1);
const usernameView = new Uint8Array(buffer, 4, 16);
const amountDueView = new Float32Array(buffer, 20, 1);

现在你就可以通过 amountDueView[0] 的方式访问数据。

备注: C 语言结构体的数据对齐与平台相关。因此需要防范和考虑不同平台字节填充对齐的差异。

转换为普通数组

在处理完一个类型化数组后,有时需要把它转为普通数组,以便可以从 Array 原型中受益。可以使用 Array.from() 实现:

const typedArray = new Uint8Array([1, 2, 3, 4]);
const normalArray = Array.from(typedArray);

也可以使用展开语法

const typedArray = new Uint8Array([1, 2, 3, 4]);
const normalArray = [...typedArray];

如果不支持 Array.from(),还可以使用以下代码。

const typedArray = new Uint8Array([1, 2, 3, 4]);
const normalArray = Array.prototype.slice.call(typedArray);

参见