复习ES6(六):数组的扩展
扩展运算符
扩展运算符(spread)是三个点(…)。它好比 rest 参数的逆运算,将一个数组转为用逗号分隔的参数序列
1 | console.log(...[1, 2, 3]) |
该运算符主要用于函数调用
扩展运算符后面还可以放置表达式
如果扩展运算符后面是一个空数组,则不产生任何效果。
只有函数调用时,扩展运算符才可以放在圆括号中,否则会报错
替代函数的apply方法
1 | // ES5 的写法 |
1 | // ES5的 写法 |
push可以多条值,使用…扩展运算符
复制数组
1 | const a1 = [1, 2]; |
a1会返回原数组的克隆,再修改a2就不会对a1产生影响
合并数组
1 | const arr1 = ['a', 'b']; |
合并数组都是浅拷贝,使用的时候需要注意。如果修改了引用指向的值,会同步反映到新数组
与解构相结合
1 | const [first, ...rest] = [1, 2, 3, 4, 5]; |
字符串
扩展运算符还可以将字符串转为真正的数组
1 | [...'hello'] |
JavaScript 会将四个字节的 Unicode 字符,识别为 2 个字符,采用扩展运算符就没有这个问题。
实现了 Iterator 接口的对象
任何定义了遍历器(Iterator)接口的对象(参阅 Iterator 一章),都可以用扩展运算符转为真正的数组
Map 和 Set 结构,Generator 函数
1 | let map = new Map([ |
Array.from()
Array.from方法用于将两类对象转为真正的数组:类似数组的对象(array-like object)和可遍历(iterable)的对象(包括 ES6 新增的数据结构 Set 和 Map)
1 | let arrayLike = { |
如果参数是一个真正的数组,Array.from会返回一个一模一样的新数组
扩展运算符背后调用的是遍历器接口(Symbol.iterator),如果一个对象没有部署这个接口,就无法转换。
Array.from方法还支持类似数组的对象。所谓类似数组的对象,本质特征只有一点,即必须有length属性。
Array.from还可以接受第二个参数,作用类似于数组的map方法
1 | Array.from(arrayLike, x => x * x); |
返回各种数据的数据类型
1 | function typesOf () { |
Array.from()的另一个应用是,将字符串转为数组,然后返回字符串的长度
Array.of()
Array.of()方法用于将一组值,转换为数组
1 | Array.of(3, 11, 8) // [3,11,8] |
arr.copyWithin()
数组实例的copyWithin()方法,在当前数组内部,将指定位置的成员复制到其他位置(会覆盖原有成员),然后返回当前数组
1 | Array.prototype.copyWithin(target, start = 0, end = this.length) |
target(必需):从该位置开始替换数据。如果为负值,表示倒数。
start(可选):从该位置开始读取数据,默认为 0。如果为负值,表示从末尾开始计算。
end(可选):到该位置前停止读取数据,默认等于数组长度。如果为负值,表示从末尾开始计算。
1 | [1, 2, 3, 4, 5].copyWithin(0, 3) |
上面代码表示将从 3 号位直到数组结束的成员(4 和 5),复制到从 0 号位开始的位置,结果覆盖了原来的 1 和 2。
arr.find() 和arr.findIndex()
数组实例的find方法,用于找出第一个符合条件的数组成员。它的参数是一个回调函数
1 | [1, 5, 10, 15].find(function(value, index, arr) { |
数组实例的findIndex方法的用法与find方法非常类似,返回第一个符合条件的数组成员的位置
1 | [1, 5, 10, 15].findIndex(function(value, index, arr) { |
arr.fill()
fill方法使用给定值,填充一个数组
1 | ['a', 'b', 'c'].fill(7, 1, 2) |
arr.entries()/arr.keys()/arr.values()
keys()是对键名的遍历、values()是对键值的遍历,entries()是对键值对的遍历
1 | for (let index of ['a', 'b'].keys()) { |
arr.includes()
Array.prototype.includes方法返回一个布尔值,表示某个数组是否包含给定的值,与字符串的includes方法类似
1 | [1, 2, 3].includes(2) // true |
arr.flat() / arr.flatMap()
Array.prototype.flat()用于将嵌套的数组“拉平”,变成一维的数组。该方法返回一个新数组,对原数据没有影响
1 | [1, 2, [3, [4, 5]]].flat() |
flatMap()方法对原数组的每个成员执行一个函数(相当于执行Array.prototype.map()),然后对返回值组成的数组执行flat()方法。该方法返回一个新数组,不改变原数组
1 | // 相当于 [[2, 4], [3, 6], [4, 8]].flat() |
flatMap()只能展开一层数组
数组中的空位
forEach(), filter(), reduce(), every() 和some()都会跳过空位。
map()会跳过空位,但会保留这个值
join()和toString()会将空位视为undefined,而undefined和null会被处理成空字符串。
1 | // forEach方法 |
ES6 则是明确将空位转为undefined。
Array.from … copyWithin() fill() for..of 会识别空位
entries()、keys()、values()、find()和findIndex()会将空位处理成undefined。
Array.prototype.sort() 的排序稳定性
1 | const arr = [ |