你不懂JS:ES6与未来
第五章:集合
结构化的集合与数据访问对于任何JS程序来说都是一个关键组成部分。从这门语言的最开始到现在,数组和对象一直都是我们创建数据结构的主要机制。当然,许多更高级的数据结构作为用户方的库都曾建立在这些之上。
到了ES6,最有用(而且优化性能的!)的数据结构抽象中的一些已经作为这门语言的原生组件被加入了进来。
我们将通过检视 类型化数组(TypedArrays) 来开始这一章,技术上讲它与几年前的ES5是同一时期的产物,但是仅仅作为WebGL的同伴被标准化了,而不是作为JavaScript自身的一部分。到了ES6,这些东西已经被语言规范直接采纳了,这给予了它们头等的地位。
Map就像对象(键/值对),但是与仅能使用一个字符串作为键不同的是,你可以使用任何值 —— 即使是另一个对象或map!Set与数组很相似(值的列表),但是这些值都是唯一的;如果你添加一个重复的值,它会被忽略。还有与之相对应的weak结构(与内存/垃圾回收有关联):WeakMap和WeakSet。
类型化数组(TypedArrays)
正如我们在本系列的 类型与文法 中讲到过的,JS确实拥有一组内建类型,比如number
和string
。看到一个称为“类型化的数组”的特性,可能会诱使你推测它意味着一个特定类型的值的数组,比如一个仅含字符串的数组。
然而,类型化数组其实更多的是关于使用类似数组的语义(索引访问,等等)提供对二进制数据的结构化访问。名称中的“类型”指的是在大量二进制位(比特桶)的类型之上覆盖的“视图”,它实质上是一个映射,控制着这些二进制位是否应当被看作8位有符号整数的数组,还是被看作16位有符号整数的数组,等等。
你怎样才能构建这样的比特桶呢?它被称为一个“缓冲(buffer)”,而你可以用ArrayBuffer(..)
构造器直接地构建它:
var buf = new ArrayBuffer( 32 );
buf.byteLength; // 32
现在buf
是一个长度为32字节(256比特)的二进制缓冲,它被预初始化为全0
。除了检查它的byteLength
属性,一个缓冲本身不会允许你进行任何操作。
提示: 有几种web平台特性都使用或返回缓冲,比如FileReader#readAsArrayBuffer(..)
,XMLHttpRequest#send(..)
,和ImageData
(canvas数据)。
但是在这个数组缓冲的上面,你可以平铺一层“视图”,它就是用类型化数组的形式表现的。考虑如下代码:
var arr = new Uint16Array( buf );
arr.length; // 16
arr
是一个256位的buf
缓冲在16位无符号整数的类型化数组的映射,意味着你得到16个元素。
字节顺序
明白一个事实非常重要:arr
是使用JS所运行的平台的字节顺序设定(大端法或小端法)被映射的。如果二进制数据是由一种字节顺序创建,但是在一个拥有相反字节数序的平台被解释时,这就可能是个问题。
字节顺序指的是一个多字节数字的低位字节(8个比特位的集合) —— 比如我们在早先的代码段中创建的16位无符号整数 —— 是在这个数字的字节序列的左边还是右边。
举个例子,让我们想象一下用16位来表示的10进制的数字3085
。如果你只有一个16位数字的容器,无论字节顺序怎样它都将以二进制表示为0000110000001101
(十六进制的0c0d
)。
但是如果3085
使用两个8位数字来表示的话,字节顺序就像会极大地影响它在内存中的存储:
0000110000001101
/0c0d
(大端法)0000110100001100
/0d0c
(小端法)
如果你从一个小端法系统中收到表示为0000110100001100
的3085
,但是在一个大端法系统中为它上面铺一层视图,那么你将会看到值3340
(10进制)和0d0c
(16进制)。
如今在web上最常见的表现形式是小端法,但是绝对存在一些与此不同的浏览器。你明白一块二进制数据的生产者和消费者的字节顺序是十分重要的。
在MDN上有一种快速的方法测试你的JavaScript的字节顺序:
var littleEndian = (function() {
var buffer = new ArrayBuffer( 2 );
new DataView( buffer ).setInt16( 0, 256, true );
return new Int16Array( buffer )[0] === 256;
})();
littleEndian
将是true
或false
;对大多数浏览器来说,它应当返回true
。这个测试使用DataView(..)
,它允许更底层,更精细地控制如何从你平铺在缓冲上的视图中访问二进制位。前面代码段中的setInt16(..)
方法的第三个参数告诉DataView
,对于这个操作你想使用什么字节顺序。
警告: 不要将一个数组缓冲中底层的二进制存储的字节顺序与一个数字在JS程序中被暴露时如何被表示搞混。举例来说,(3085).toString(2)
返回"110000001101"
,它被假定前面有四个"0"
因而是大端法表现形式。事实上,这个表现形式是基于一个单独的16位视图的,而不是两个8位字节的视图。上面的DataView
测试是确定你的JS环境的字节顺序的最佳方法。
多视图
一个单独的缓冲可以连接多个视图,例如:
var buf = new ArrayBuffer( 2 );
var view8 = new Uint8Array( buf );
var view16 = new Uint16Array( buf );
view16[0] = 3085;
view8[0]; // 13
view8[1]; // 12
view8[0].toString( 16 ); // "d"
view8[1].toString( 16 ); // "c"
// 调换(好像字节顺序一样!)
var tmp = view8[0];
view8[0] = view8[1];
view8[1] = tmp;
view16[0]; // 3340
类型化数组的构造器拥有多种签名。目前我们展示过的只是向它们传递一个既存的缓冲。然而,这种形式还接受两个额外的参数:byteOffset
和length
。换句话讲,你可以从0
以外的位置开始类型化数组视图,也可以使它的长度小于整个缓冲的长度。
如果二进制数据的缓冲包含规格不一的大小/位置,这种技术可能十分有用。
例如,考虑一个这样的二进制缓冲:在开头拥有一个2字节数字(也叫做“字”),紧跟着两个1字节数字,然后跟着一个32位浮点数。这是你如何在同一个缓冲,偏移量,和长度上使用多视图来访问数据:
var first = new Uint16Array( buf, 0, 2 )[0],
second = new Uint8Array( buf, 2, 1 )[0],
third = new Uint8Array( buf, 3, 1 )[0],
fourth = new Float32Array( buf, 4, 4 )[0];
类型化数组构造器
除了前一节我们检视的(buffer,[offset, [length]])
形式之外,类型化数组的构造器还支持这些形式:
- [constructor]
(length)
:在一个长度为length
字节的缓冲上创建一个新视图 - [constructor]
(typedArr)
:创建一个新视图和缓冲,并拷贝typedArr
视图中的内容 - [constructor]
(obj)
:创建一个新视图和缓冲,并迭代类数组或对象obj
来拷贝它的内容
在ES6中可以使用下面的类型化数组构造器:
Int8Array
(8位有符号整数),Uint8Array
(8位无符号整数)Uint8ClampedArray
(8位无符号整数,每个值都被卡在0 - 255
范围内)
Int16Array
(16位有符号整数),Uint16Array
(16位无符号整数)Int32Array
(32位有符号整数),Uint32Array
(32位无符号整数)Float32Array
(32位浮点数,IEEE-754)Float64Array
(64位浮点数,IEEE-754)
类型化数组构造器的实例基本上和原生的普通数组是一样的。一些区别包括它有一个固定的长度并且值都是同种“类型”。
但是,它们共享绝大多数相同的prototype
方法。这样一来,你很可能将会像普通数组那样使用它们而不必进行转换。
例如:
var a = new Int32Array( 3 );
a[0] = 10;
a[1] = 20;
a[2] = 30;
a.map( function(v){
console.log( v );
} );
// 10 20 30
a.join( "-" );
// "10-20-30"
警告: 你不能对类型化数组使用没有意义的特定Array.prototype
方法,比如修改器(splice(..)
,push(..)
,等等)和concat(..)
。
要小心,在类型化数组中的元素被限制在它被声明的位长度中。如果你有一个Uint8Array
并试着向它的一个元素赋予某些大于8为的值,那么这个值将被截断以保持在相应的位长度中。
这可能造成一些问题,例如,如果你试着对一个类型化数组中的所有值求平方。考虑如下代码:
var a = new Uint8Array( 3 );
a[0] = 10;
a[1] = 20;
a[2] = 30;
var b = a.map( function(v){
return v * v;
} );
b; // [100, 144, 132]
在被平方后,值20
和30
的结果会位溢出。要绕过这样的限制,你可以使用TypedArray#from(..)
函数:
var a = new Uint8Array( 3 );
a[0] = 10;
a[1] = 20;
a[2] = 30;
var b = Uint16Array.from( a, function(v){
return v * v;
} );
b; // [100, 400, 900]
关于被类型化数组所共享的Array.from(..)
函数的更多信息,参见第六章的“Array.from(..)
静态方法”一节。特别地,“映射”一节讲解了作为第二个参数值被接受的映射函数。
一个值得考虑的有趣行为是,类型化数组像普通数组一样有一个sort(..)
方法,但是这个方法默认是数字排序比较而不是将值强制转换为字符串进行字典顺序比较。例如:
var a = [ 10, 1, 2, ];
a.sort(); // [1,10,2]
var b = new Uint8Array( [ 10, 1, 2 ] );
b.sort(); // [1,2,10]
就像Array#sort(..)
一样,TypedArray#sort(..)
接收一个可选的比较函数作为参数值,它们的工作方式完全一样。
Maps
如果你对JS经验丰富,那么你一定知道对象是创建无序键/值对数据结构的主要机制,这也被称为map。然而,将对象作为map的主要缺陷是不能使用一个非字符串值作为键。
例如,考虑如下代码:
var m = {};
var x = { id: 1 },
y = { id: 2 };
m[x] = "foo";
m[y] = "bar";
m[x]; // "bar"
m[y]; // "bar"
这里发生了什么?x
和y
这两个对象都被字符串化为"[object Object]"
,所以只有这一个键被设置为m
。
一些人通过在一个值的数组旁边同时维护一个平行的非字符串键的数组实现了山寨的map,比如:
var keys = [], vals = [];
var x = { id: 1 },
y = { id: 2 };
keys.push( x );
vals.push( "foo" );
keys.push( y );
vals.push( "bar" );
keys[0] === x; // true
vals[0]; // "foo"
keys[1] === y; // true
vals[1]; // "bar"
当然,你不会想亲自管理这些平行数组,所以你可能会定义一个数据解构,使它内部带有自动管理的方法。除了你不得不自己做这些工作,主要的缺陷是访问的时间复杂度不再是O(1),而是O(n)。
但在ES6中,不再需要这么做了!使用Map(..)
就好:
var m = new Map();
var x = { id: 1 },
y = { id: 2 };
m.set( x, "foo" );
m.set( y, "bar" );
m.get( x ); // "foo"
m.get( y ); // "bar"
唯一的缺点是你不能使用[]
方括号访问语法来设置或取得值。但是get(..)
和set(..)
可以完美地取代这种语法。
要从一个map中删除一个元素,不要使用delete
操作符,而是使用delete(..)
方法:
m.set( x, "foo" );
m.set( y, "bar" );
m.delete( y );
使用clear()
你可清空整个map的内容。要得到map的长度(也就是,键的数量),使用size
属性(不是length
)。
m.set( x, "foo" );
m.set( y, "bar" );
m.size; // 2
m.clear();
m.size; // 0
Map(..)
的构造器还可以接受一个可迭代对象(参见第三章的“迭代器”),它必须产生一个数组的列表,每个数组的第一个元素是键,第二元素是值。这种用于迭代的格式与entries()
方法产生的格式是一样的,entries()
方法将在下一节中讲解。这使得制造一个map的拷贝十分简单:
var m2 = new Map( m.entries() );
// 等同于:
var m2 = new Map( m );
因为一个map实例是一个可迭代对象,而且它的默认迭代器与entries()
相同,第二种稍短的形式更理想。
当然,你可以在Map(..)
构造器形式中手动指定一个 entries 列表:
var x = { id: 1 },
y = { id: 2 };
var m = new Map( [
[ x, "foo" ],
[ y, "bar" ]
] );
m.get( x ); // "foo"
m.get( y ); // "bar"
Map 值
要从一个map得到值的列表,使用values(..)
,它返回一个迭代器。在第二和第三章,我们讲解了几种序列化(像一个数组那样)处理一个迭代器的方法,比如...
扩散操作符和for..of
循环。另外,第六章的“Arrays”将会详细讲解Array.from(..)
方法。考虑如下代码:
var m = new Map();
var x = { id: 1 },
y = { id: 2 };
m.set( x, "foo" );
m.set( y, "bar" );
var vals = [ ...m.values() ];
vals; // ["foo","bar"]
Array.from( m.values() ); // ["foo","bar"]
就像在前一节中讨论过的,你可以使用entries()
(或者默认的map迭代器)迭代一个map的记录。考虑如下代码:
var m = new Map();
var x = { id: 1 },
y = { id: 2 };
m.set( x, "foo" );
m.set( y, "bar" );
var vals = [ ...m.entries() ];
vals[0][0] === x; // true
vals[0][1]; // "foo"
vals[1][0] === y; // true
vals[1][1]; // "bar"
Map 键
要得到键的列表,使用keys()
,它返回一个map中键的迭代器:
var m = new Map();
var x = { id: 1 },
y = { id: 2 };
m.set( x, "foo" );
m.set( y, "bar" );
var keys = [ ...m.keys() ];
keys[0] === x; // true
keys[1] === y; // true
要判定一个map中是否拥有一个给定的键,使用has(..)
:
var m = new Map();
var x = { id: 1 },
y = { id: 2 };
m.set( x, "foo" );
m.has( x ); // true
m.has( y ); // false
实质上map让你将一些额外的信息(值)与一个对象(键)相关联,而不用实际上将这些信息放在对象本身中。
虽然在一个map中你可以使用任意种类的值作为键,但是你经常使用的将是对象,就像字符串和其他在普通对象中可以合法地作为键的基本类型。换句话说,你可能将想要继续使用普通对象,除非一些或全部的键需要是对象,在那种情况下map更合适。
警告: 如果你使用一个对象作为一个map键,而且这个对象稍后为了能够被垃圾回收器(GC)回收它占用的内存而被丢弃(解除所有的引用),那么map本身将依然持有它的记录。你需要从map中移除这个记录来使它能够被垃圾回收。在下一节中,我们将看到对于作为对象键和GC来说更好的选择 —— WeakMaps。
WeakMaps
WeakMap是map的一个变种,它们的大多数外部行为是相同的,而在底层内存分配(明确地说是它的GC)如何工作上有区别。
WeakMap(仅)接收对象作为键。这些对象被 弱 持有,这意味着如果对象本身被垃圾回收掉了,那么在WeakMap中的记录也会被移除。这是观察不到的,因为一个对象可以被垃圾回收的唯一方法是不再有指向它的引用 —— 一旦不再有指向它的引用,你就没有对象引用可以用来检查它是否存在于这个WeakMap中。
除此以外,WeakMap的API是相似的,虽然限制更多:
var m = new WeakMap();
var x = { id: 1 },
y = { id: 2 };
m.set( x, "foo" );
m.has( x ); // true
m.has( y ); // false
WeakMap没有size
属性和clear()
方法,它们也不对它们的键,值和记录暴露任何迭代器。所以即便你解除了x
引用,它将会因GC从m
中移除它的记录,也没有办法确定这一事实。你只能相信JavaScript会这么做!
就像map一样,WeakMap让你将信息与一个对象软关联。如果你不能完全控制这个对象,比如DOM元素,它们就特别有用。如果你用做map键的对象可以被删除并且应当在被删除时成为GC的回收对象,那么一个WeakMap就是更合适的选项。
要注意的是WeakMap只弱持有它的 键,而不是它的值。考虑如下代码:
var m = new WeakMap();
var x = { id: 1 },
y = { id: 2 },
z = { id: 3 },
w = { id: 4 };
m.set( x, y );
x = null; // { id: 1 } 是可以GC的
y = null; // 由于 { id: 1 } 是可以GC的,因此 { id: 2 } 也可以
m.set( z, w );
w = null; // { id: 4 } 不可以GC
因此,我认为WeakMap被命名为“WeakKeyMap”更好。
Sets
一个set是一个集合,其中的值都是唯一的(重复的会被忽略)。
set的API与map很相似。add(..)
方法(有点讽刺地)取代了set(..)
,而且没有get(..)
方法。
考虑如下代码:
var s = new Set();
var x = { id: 1 },
y = { id: 2 };
s.add( x );
s.add( y );
s.add( x );
s.size; // 2
s.delete( y );
s.size; // 1
s.clear();
s.size; // 0
Set(..)
构造器形式与Map(..)
相似,它可以接收一个可迭代对象,比如另一个set或者一个值的数组。但是,与Map(..)
期待一个 记录 的列表(键/值数组的数组)不同的是,Set(..)
期待一个 值 的列表(值的数组):
var x = { id: 1 },
y = { id: 2 };
var s = new Set( [x,y] );
一个set不需要get(..)
,因为你不会从一个set中取得值,而是使用has(..)
测试一个值是否存在:
var s = new Set();
var x = { id: 1 },
y = { id: 2 };
s.add( x );
s.has( x ); // true
s.has( y ); // false
注意: has(..)
中的比较算法与Object.is(..)
(见第六章)几乎完全相同,除了-0
和0
被视为相同而非不同。
Set 迭代器
set和map一样拥有相同的迭代器方法。set的行为有所不同,但是与map的迭代器的行为是对称的。考虑如下代码:
var s = new Set();
var x = { id: 1 },
y = { id: 2 };
s.add( x ).add( y );
var keys = [ ...s.keys() ],
vals = [ ...s.values() ],
entries = [ ...s.entries() ];
keys[0] === x;
keys[1] === y;
vals[0] === x;
vals[1] === y;
entries[0][0] === x;
entries[0][1] === x;
entries[1][0] === y;
entries[1][1] === y;
keys()
和values()
迭代器都会给出set中唯一值的列表。entries()
迭代器给出记录数组的列表,记录数组中的两个元素都是唯一的set值。一个set的默认迭代器是它的values()
迭代器。
一个set天生的唯一性是它最有用的性质。例如:
var s = new Set( [1,2,3,4,"1",2,4,"5"] ),
uniques = [ ...s ];
uniques; // [1,2,3,4,"1","5"]
set的唯一性不允许强制转换,所以1
和"1"
被认为是不同的值。
WeakSets
一个WeakMap弱持有它的键(但强持有它的值),而一个WeakSet弱持有它的值(不存在真正的键)。
var s = new WeakSet();
var x = { id: 1 },
y = { id: 2 };
s.add( x );
s.add( y );
x = null; // `x` 可以GC
y = null; // `y` 可以GC
警告: WeakSet的值必须是对象,在set中被允许的基本类型值是不行的。
复习
ES6定义了几种有用的集合,它们使得处理解构化的数据更加高效和有效。
类型化数组提供了二进制数据缓冲的“视图”,它使用各种整数类型对齐,比如8位无符号整数和32位浮点数。二进制数据的数组访问使得操作更加容易表达和维护,它可以让你更简单地处理如视频,音频,canvas数据等复杂的数组。
Map是键-值对集合,它的键可以是对象而非只可以是字符串/基本类型。Set是(任何类型的)唯一值的列表。
WeakMap是键(对象)被弱持有的map,所以如果它是最后一个指向这个对象的引用,GC就可以自由地回收这个记录。WeakSet是值被弱持有的set,所以同样地,如果它是最后一个指向这个对象的引用,GC就可以移除这个记录。