1
0
mirror of https://github.com/ruanyf/es6tutorial.git synced 2025-05-24 18:32:22 +00:00
es6tutorial/docs/array.md
2014-06-12 14:17:54 +08:00

235 lines
5.9 KiB
Markdown
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

# 数组的扩展
## Array.from()
Array.from()用于将两类对象转为真正的数组类似数组的对象array-like object和可遍历iterable的对象其中包括ES6新增的Set和Map结构。
```javascript
let ps = document.querySelectorAll('p');
Array.from(ps).forEach(function (p) {
console.log(p);
});
```
上面代码中querySelectorAll方法返回的是一个类似数组的对象只有将这个对象转为真正的数组才能使用forEach方法。
Array.from()还可以接受第二个参数作用类似于数组的map方法用来对每个元素进行处理。
```JavaScript
Array.from(arrayLike, x => x * x);
// 等同于
Array.from(arrayLike).map(x => x * x);
```
## Array.of()
Array.of()方法用于将一组值,转换为数组。
```javaScript
Array.of(3, 11, 8) // [3,11,8]
Array.of(3).length // 1
```
这个函数的主要目的是弥补数组构造函数Array()的不足。因为参数个数的不同会导致Array()的行为有差异。
```javascript
Array() // []
Array(3) // [undefined, undefined, undefined]
Array(3,11,8) // [3, 11, 8]
```
上面代码说明只有当参数个数不少于2个Array()才会返回由参数组成的新数组。
## 数组实例的find()和findIndex()
数组实例的find()用于找出第一个符合条件的数组元素。它的参数是一个回调函数所有数组元素依次遍历该回调函数直到找出第一个返回值为true的元素然后返回该元素否则返回undefined。
```javascript
[1, 5, 10, 15].find(function(value, index, arr) {
return value > 9;
}) // 10
```
从上面代码可以看到,回调函数接受三个参数,依次为当前的值、当前的位置和原数组。
数组实例的findIndex()的用法与find()非常类似,返回第一个符合条件的数组元素的位置,如果所有元素都不符合条件,则返回-1。
```javascript
[1, 5, 10, 15].findIndex(function(value, index, arr) {
return value > 9;
}) // 2
```
这两个方法都可以接受第二个参数用来绑定回调函数的this对象。
另外这两个方法都可以发现NaN弥补了IndexOf()的不足。
```javascript
[NaN].indexOf(NaN)
// -1
[NaN].findIndex(y => Object.is(NaN, y))
// 0
```
## 数组实例的fill()
fill()使用给定值,填充一个数组。
```javascript
['a', 'b', 'c'].fill(7)
// [7, 7, 7]
new Array(3).fill(7)
// [7, 7, 7]
```
上面代码表明fill方法用于空数组的初始化非常方便。数组中已有的元素会被全部抹去。
fill()还可以接受第二个和第三个参数,用于指定填充的起始位置和结束位置。
```javascript
['a', 'b', 'c'].fill(7, 1, 2)
// ['a', 7, 'c']
```
## 数组实例的entries()keys()和values()
ES6提供三个新的方法——entries()keys()和values()——用于遍历数组。它们都返回一个遍历器可以用for...of循环进行遍历唯一的区别是keys()是对键名的遍历、values()是对键值的遍历entries()是对键值对的遍历。
```javascript
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"
```
## 数组推导
ES6提供简洁写法允许直接通过现有数组生成新数组这被称为数组推导array comprehension
```javascript
var a1 = [1, 2, 3, 4];
var a2 = [for (i of a1) i * 2];
a2 // [2, 4, 6, 8]
```
上面代码表示通过for...of结构数组a2直接在a1的基础上生成。
注意数组推导中for...of结构总是写在最前面返回的表达式写在最后面。
for...of后面还可以附加if语句用来设定循环的限制条件。
```javascript
var years = [ 1954, 1974, 1990, 2006, 2010, 2014 ];
[for (year of years) if (year > 2000) year];
// [ 2006, 2010, 2014 ]
[for (year of years) if (year > 2000) if(year < 2010) year];
// [ 2006]
[for (year of years) if (year > 2000 && year < 2010) year];
// [ 2006]
```
上面代码表明if语句写在for...of与返回的表达式之间可以使用多个if语句。
数组推导可以替代map和filter方法。
```javascript
[for (i of [1, 2, 3]) i * i];
// 等价于
[1, 2, 3].map(function (i) { return i * i });
[for (i of [1,4,2,3,-8]) if (i < 3) i];
// 等价于
[1,4,2,3,-8].filter(function(i) { return i < 3 });
```
上面代码说明模拟map功能只要单纯的for...of循环就行了模拟filter功能除了for...of循环还必须加上if语句。
在一个数组推导中还可以使用多个for...of结构构成多重循环。
```javascript
var a1 = ["x1", "y1"];
var a2 = ["x2", "y2"];
var a3 = ["x3", "y3"];
[for (s of a1) for (w of a2) for (r of a3) console.log(s + w + r)];
// x1x2x3
// x1x2y3
// x1y2x3
// x1y2y3
// y1x2x3
// y1x2y3
// y1y2x3
// y1y2y3
```
上面代码在一个数组推导之中使用了三个for...of结构。
需要注意的是数组推导的方括号构成了一个单独的作用域在这个方括号中声明的变量类似于使用let语句声明的变量。
由于字符串可以视为数组,因此字符串也可以直接用于数组推导。
```javascript
[for (c of 'abcde') if (/[aeiou]/.test(c)) c].join('') // 'ae'
[for (c of 'abcde') c+'0'].join('') // 'a0b0c0d0e0'
```
上面代码使用了数组推导,对字符串进行处理。
数组推导需要注意的地方是,新数组会立即在内存中生成。这时,如果原数组是一个很大的数组,将会非常耗费内存。
## Array.observe()Array.unobserve()
这两个方法用于监听(取消监听)数组的变化,指定回调函数。
它们的用法与Object.observe和Object.unobserve方法完全一致也属于ES7的一部分请参阅《对象的扩展》一章。唯一的区别是对象可监听的变化一共有六种而数组只有四种add、update、delete、splice数组的length属性发生变化