共计 1394 个字符,预计需要花费 4 分钟才能阅读完成。
1. 什么是伪数组
JavaScript 中存在有一品种数组,或者说伪数组。常常见到的伪数组有函数的 arguments
对象、dom.querySelectorAll
等获取的 NodeList
类(NodeList
自身具备 forEach
办法)等。
伪数组并不是数组,它没有继承Array.prototype
, 然而它“看起来像数组”,它自身没有数组的规范办法,然而它能够复用这些规范办法。
例子
function arrayLike() {arguments.forEach(a => console.log(a));//TypeError: arguments.forEach is not a function
}
arrayLike(1, 2, 3);
如上例所示,arguments
对象自身并没有 forEach
办法,然而它能够复用数组的这些规范办法。
例子
function arrayLike() {// arguments.forEach(a => console.log(a));
[].forEach.call(arguments, a => console.log(a));// 1 2 3 通过 call 扭转 this 指向,调用数组的办法
[...arguments].forEach(a => console.log(a));// 1 2 3 构建一个实在数组,而后调用数组的办法
}
arrayLike(1, 2, 3);
2. 如何创立一个伪数组对象
一个数组对象必然具备两个特点:
- 具备一个范畴在 0~232-1 的整型 length 属性
- length 属性大于该对象的最大索引,索引是一个 0-232 -2 范畴内的整数
所以很简略,只有实现这两个特点,一个对象就是伪数组对象了。
例子
const arr = {
1: 'AAA',
3: 'CCC',
length: 8,
};
[].forEach.call(arr, (item, i) => console.log(item, i)); //AAA 1 CCC 3
3. 数组的 concat
办法
对于数组和伪数组,在数组的规范办法中,只有 concat
办法是不通用的,对于一个伪数组,concat
办法会将其作为一个整体连接起来。
例子
console.log([].concat.call(arr, [7, 8]));//[{ '1': 'AAA', '3': 'CCC', length: 8}, 7, 8 ]
console.log([1, 2].concat([7, 8]));//[1, 2, 7, 8]
上例展现了数组和伪数组调用 concat
的不同后果,在遇到这种状况时,咱们只有本人对伪数组进行转换,比方:
1. 通过 slice 办法,复制伪数组
console.log([].concat.call([].slice.call(arr), [7, 8]));
//[<1 empty item>, 'AAA', <1 empty item>, 'CCC', <4 empty items>, 7, 8]
2. 通过 Symbol.isConcatSpreadable
扭转对伪数组对象进行 concat
操作时的默认行为
const arr = {
1: 'AAA',
3: 'CCC',
length: 8,
[Symbol.isConcatSpreadable]: true,
};
console.log([].concat.call(arr, [7, 8]));
//[<1 empty item>, 'AAA', <1 empty item>, 'CCC', <4 empty items>, 7, 8]
正文完