ES6学习笔记:数组的扩展,ES6为各种内置类型增加了一些静态方法和原型方法。
Array
静态方法
Array.from()
Array.from方法用于将两类对象转为真正的数组:
类似数组的对象 可遍历的对象(包括ES6新增的数据结构Set和Map)。下面是一个类似数组的对象,Array.from将它转为真正的数组。
let arrayLike = { '0': 'a', '1': 'b', '2': 'c', length: 3 }; // ES5的写法 var arr1 = [].slice.call(arrayLike); // ['a', 'b', 'c'] // ES6的写法 let arr2 = Array.from(arrayLike); // ['a', 'b', 'c']
实际应用中,常见的类似数组的对象是DOM操作返回的NodeList集合,以及函数内部的arguments对象。Array.from都可以将它们转为真正的数组。
// NodeList对象 let ps = document.querySelectorAll('p'); Array.from(ps).forEach(function (p) { console.log(p); }); // arguments对象 function foo() { var args = Array.from(arguments); // ... }
只要是部署了Iterator接口的数据结构,Array.from都能将其转为数组:
Array.from('hello') // ['h', 'e', 'l', 'l', 'o'] let namesSet = new Set(['a', 'b']) Array.from(namesSet) // ['a', 'b']
上面代码中,字符串和Set结构都具有Iterator接口,因此可以被Array.from转为真正的数组。
如果参数是一个真正的数组,Array.from会返回一个一模一样的新数组:
Array.from([1, 2, 3]) // [1, 2, 3]
扩展运算符(…)也可以将某些数据结构转为数组。
// arguments对象 function foo() { var args = [...arguments]; } // NodeList对象 [...document.querySelectorAll('p')]扩展运算符背后调用的是遍历器接口(Symbol.iterator),如果一个对象没有部署这个接口,就无法转换 Array.from方法则是还支持类似数组的对象。所谓类似数组的对象,本质特征只有一点,即必须有length属性。因此,任何有length属性的对象,都可以通过Array.from方法转为数组,而此时扩展运算符就无法转换。
Array.from({ length: 3 }); // [ undefined, undefined, undefined ]
polyfill:
if(!Array.from){ Array.from = function(obj){ [].slice().call(obj); } }
Array.from还可以接受第二个参数,作用类似于数组的map方法,用来对每个元素进行处理,将处理后的值放入返回的数组。
Array.from(arrayLike, x => x * x); // 等同于 Array.from(arrayLike).map(x => x * x); Array.from([1, 2, 3], (x) => x * x) // [1, 4, 9]
Array.from([1, , 2, , 3], (n) => n || 0); // [1, 0, 2, 0, 3]
如果map函数里面用到了this关键字,还可以传入Array.from的第三个参数,用来绑定this。
Array.from()的另一个应用是,将字符串转为数组,然后返回字符串的长度。因为它能正确处理各种Unicode字符,可以避免JavaScript将大于\uFFFF的Unicode字符,算作两个字符的bug。
function countSymbols(string) { return Array.from(string).length; }
Array.of()
Array.of方法用于将一组值,转换为数组。
Array.of(3,4,5); //[3,4,5] Array.of(3) // [3] Array.of(3).length // 1
这个方法的主要目的,是弥补数组构造函数Array()的不足。因为参数个数的不同,会导致Array()的行为有差异。
Array() // [] Array(3) // [, , ,] Array(3, 11, 8) // [3, 11, 8]
Array的参数个数只有一个的时候,实际上是指定数组的长度。
Array.of基本上可以用来替代Array()或new Array(),并且不存在由于参数不同而导致的重载。它的行为非常统一。
Array.of() // [] Array.of(undefined) // [undefined] Array.of(1) // [1] Array.of(1, 2) // [1, 2]
Array.of总是返回参数值组成的数组。如果没有参数,就返回一个空数组。
polyfill:
if(!Array.of){ Array.of = function(){ return [].slice().call(arguments); } }
实例方法
copyWithin()
数组实例的copyWithin方法,在当前数组内部,将指定位置的成员复制到其他位置(会覆盖原有成员),然后返回当前数组。
也就是说,使用这个方法,会修改当前数组。
Array.prototype.copyWithin(target, start = 0, end = this.length)
它接受三个参数:
target(必需):从该位置开始替换数据。 start(可选):从该位置开始读取数据,默认为0。如果为负值,表示从末尾开始算。 end(可选):到该位置前停止读取数据,不包括该位置的数据。默认等于数组长度。如果为负值,表示从末尾开始算。[1,2,3,4,5].copyWithin(-2); //[1,2,3,1,2] [1,2,3,4,5].copyWithin(0,3); //[4,5,3,4,5] [1,2,3,4,5].copyWithin(0,3,4); //[4,2,3,4,5] [1,2,3,4,5].copyWithin(-2,-3,-1); //[1,2,3,3,4] // 将3号位复制到0号位 [].copyWithin.call({length: 5, 3: 1}, 0, 3) ({0:undefined,1:undefined,2:undefined,3: 1,4:undefined,5:undefined,length: 5}).copyWithin(0,3); 结果为: {0:1,1:undefined,2:undefined,3: 1,4:undefined,5:undefined,length: 5}; 也就是 {0:1,3:1,length:5}
find() findIndex()
find() 方法返回数组中满足提供的测试函数的第一个元素的值。否则返回 undefined。
function isBigEnough(element) { return element >= 15; } [12, 5, 8, 130, 44].find(isBigEnough); // 130
find()方法接受一个函数作为参数,此函数可以接收三个参数:
element:当前遍历到的元素 index:当前遍历到的索引 array:数组本身另外还可以指定第二个参数用于指定函数的this。
find方法对数组中的每一项元素执行一次callback 函数,直至有一个callback返回true。当找到了这样一个元素后,该方法会立即返回这个元素的值,否则返回undefined。
find方法不会改变数组。
[1, 5, 10, 15].find(function(value, index, arr) { return value > 9; }) // 10
数组实例的findIndex方法的用法与find方法非常类似,返回第一个符合条件的数组成员的位置,如果所有成员都不符合条件,则返回-1。
[1, 5, 10, 15].findIndex(function(value, index, arr) { return value > 9; }) // 2
如果你需要找到一个元素的位置或者一个元素是否存在于数组中,使用Array.prototype.indexOf() 或Array.prototype.includes()。
fill()
fill方法使用给定值,填充一个数组。
['a', 'b', 'c'].fill(7) // [7, 7, 7] new Array(3).fill(7) // [7, 7, 7]
上面代码表明,fill方法用于空数组的初始化非常方便。数组中已有的元素,会被全部抹去。
fill方法还可以接受第二个和第三个参数,用于指定填充的起始位置和结束位置。
['a', 'b', 'c'].fill(7, 1, 2) // ['a', 7, 'c']
entries(),keys()和values()
ES6提供三个新的方法——entries(),keys()和values()——用于遍历数组。它们都返回一个遍历器对象,可以用for…of循环进行遍历。
唯一的区别是keys()是对键名的遍历、values()是对键值的遍历,entries()是对键值对的遍历。
for (let index of ['a', 'b'].keys()) { console.log(index); } // 0 // 1 for (let elem of ['a', 'b'].values()) { console.log(elem); } // 'a' // 'b' for (let [index, elem] of ['a', 'b'].entries()) { console.log(index, elem); } // 0 "a" // 1 "b"
includes()
includes() 方法用来判断当前数组是否包含某指定的值,如果是,则返回 true,否则返回 false。
var a = [1, 2, 3]; a.includes(2); // true a.includes(4); // false
该方法的第二个参数fromIndex 表示搜索的起始位置,默认为0。
如果fromIndex 大于等于数组长度 ,则返回 false 。
[1, 2, 3].includes(3, 3); // false [1, 2, 3].includes(3, -1); // true
没有该方法之前,我们通常使用数组的indexOf方法,检查是否包含某个值。
indexOf方法有两个缺点,一是不够语义化,它的含义是找到参数值的第一个出现位置,所以要去比较是否不等于-1,表达起来不够直观。二是,它内部使用严格相当运算符(===)进行判断,这会导致对NaN的误判。
[NaN].indexOf(NaN)
includes使用的是不一样的判断算法,就没有这个问题。
[NaN].includes(NaN) // true