用最简洁的描述弄懂es6

1.let 变量声明以及特性

声明变量

  1. let a;
  2. let b, c, d;
  3. let e = 1;
  4. let f = 2, g = 3;

特性

  • 不能重复声明
  • 块级作用域 只在块级作用域有效
  • 没有变量提升  
  • 不影响作用域链

    2.const 常量声明以及特性

     特性

  • 必须有初始值

  • 一般常量使用大写
  • 常量的值不能修改
  • 也是块级作用域
  • 对于数组和对象的修改,不算对常量的修改,不会报错(可以对数组和对象进行修改,指向的地址没有改变)

3.变量的解构赋值

通过解构赋值, 可以将属性/值从对象/数组中取出,赋值给其他变量。

数组的解构

  1. const RNG = ['uzi', 'mlxg', 'letme', 'ming']
  2. let [a, b, c, d] = RNG;// 每个值就是数组的位置对应的值

对象的解构

  1. const UZI= {
  2.   name: '自豪',
  3.   age: 22,
  4.   love: 'LOL'
  5.   lol: function(){
  6.     alert('打lol')
  7.   }
  8. }
  9. let {name, age, love} = UZI;// 每个值是对象中对应的值
  10. let {lol} = UZI;
  11. lol();// 使用

4.模板字符串 ``

内容中可以直接出现换行符,单引号双引号内容不能直接用换行符(需要通过引号加号拼接)

  1. let str = `lol s10
  2.       tes 是冠军`;// 可以这样直接使用换行符
  3. 复制代码

直接进行变量的拼接  ${}

  1. let lol = 'lol十周年';
  2. let uzi = `${lol} 中国队夺冠`

5.对象的简写

  1. let name = 'zhangning187';
  2. let change = function(){
  3.   console.log('努力改变');
  4. }
  5. const supwisdom = {
  6.   name,
  7.   change
  8. }

6.箭头函数及声明特点 =>

  1. let fn = (a, b) => {
  2.   return a + b;
  3. }
  4. fn(1, 2);// 3

特点

  • 1.this是静态的,this始终是指向函数声明时所在作用域下的this值,他没有自己的this
  • 2.不能作为构造实例化对象
  • 3.不能使用arguments变量
  • 4.箭头函数的简写   

省略小括号,当形参有且只有一个的时候可以省略

  1. let add = n => {
  2.         return n*2;
  3.       }

省略花括号,当代码体只有一条语句的时候,return 必须省略,语句的执行结果就是函数的返回值

  1. let add = n => n*2;

7.函数参数的默认值

形参初始值

  1. function add(a, b, c = 10) {// 当不传递c的时候,c的默认值是10,尽量把具有默认值的参数放在后面
  2.   return a + b + c;
  3. }
  4. add (1, 2);// 13

与解构赋值结合

  1. function con({name, age, love, height = '18'}) {// 还可以给默认值
  2.   console.log(age);// 24
  3.   console.log(height);// 没有传递,使用默认值 18
  4. }
  5. con({
  6.   name: 'zhangning187',
  7.   age: 24,
  8.   love: 'js'
  9. })

8.rest参数

用于获取参数的实参,用于代替arguments
rest参数是一个数组和es5中的arguments不一样,arguments里面是一个对象
获取实参的方式

  1. function data(a, b, ...args) {// rest 参数必须放到参数的最后面
  2.     console.log(a);// 1
  3.     console.log(b);// 2
  4.     console.log(args);// [3, 4, 5, 6, 7]
  5. }
  6. data(1, 2, 3, 4, 5, 6, 7);

9.扩展运算符  …

可以将数组转换为逗号分隔的参数序列,将一个数组分割,并将各个项作为分离的参数序列传递给函数

  1. const RNG = ['UZI', 'MLXG', 'LETME', 'MING'];
  2. console.log(...RNG)// UZI MLXG LETME MING  解构之后的序列
  3. console.log(RNG);// ['UZI', 'MLXG', 'LETME', 'MING']  返回的是一个数组
  4. const a = [1,2], b=[3,6];
  5. const c = [...a, ...b];// [1, 2, 3, 6]

10.Symbol新的原始数据类型,表示独一无二的值

他是javaScript 语言的第七种数据类型,是一种类似于字符串的数据类型
  特点

  • 1.Symbol 的值是唯一的,用来解决命名冲突的问题
  • 2.Symbol 值不能与其他数据进行运算,也不能自己运算 + - * /
  • 3.Symbol 定义的对象属性不能使用for in循环遍历,但是可以使用 Reflect.ownKeys 来获取对象的所有键名
    1. //创建Symbol
    2. let s = Symbol();// s不可见,在内部实现唯一性
    3. let s2 = Symbol('zhangning187');// 这里面的字符串只是一个标志,Symbol返回的值都是唯一的
    4. let s3 = Symbol('zhangning187');
    5. console.log(s2 == s3);// false,确定唯一性
    6. //Symbol.for()方法创建,这是一个对象,这种方式可以得出唯一的Symbol值
    7. let s6 = Symbol.for('zhangning187');
    8. let s8 = Symbol.for('zhangning187');
    9. console.log(s6 ==s8);// true  得到唯一的Symbol值

对象添加Symbol类型的属性

  1. let zn = {
  2.       up: function(){},
  3.       down: function(){},
  4.       name: 'zhangning187',
  5.       age: 24
  6.     }
  7.     // 向对象zn中添加 up down 方法
  8.     // zn.up = function(){}// 这个可以添加但是不确定zn中是否存在up方法,可能会覆盖原来的up方法
  9.     // 这时候需要考虑通过Symbol添加唯一的方法
  10.     // 声明一个对象
  11.     let methods = {
  12.       up: Symbol(),
  13.       down: Symbol()
  14.     }
  15.     zn[methods.up] = function(){
  16.       console.log('我可以爬楼');
  17.     }
  18.     zn[methods.down] = function(){
  19.       console.log('我可以下楼');
  20.     }
  21.     console.log(zn);// 已经添加唯一的方法 up down
  22.     
  23.     let UZI = {
  24.       name: '自豪',
  25.       // Symbol(): function(){},// 这里不能这样直接使用, Symbol()是一个表达式,是一个动态的
  26.       [Symbol('lol')]: function(){
  27.         console.log('我会打lol');
  28.       },
  29.       [Symbol('篮球')]: function(){// Symbol()中还可以添加描述字符串
  30.         console.log('我可以打篮球')
  31.       }
  32.     }
  33.     console.log(UZI);

11.迭代器(Iterator)

迭代器是一种接口,为各种不同的数据结构提供统一的访问机制,任何数据结构只要部署Iterator接口,就可以完成遍历操作
1.ES6创造了一种新的遍历命令for…of循环,Iterator接口提供for…of消费
2.原生具备iterator接口的数据

  1. Array Arguments Set Map String TypedArray NodeList

3.工作原理

  • a 创建一个指针对象,指向当前数据结构的起始位置
  • b 第一次调用对象的next方法,指针自动指向数据结构的第一个成员
  • c 接下来不断调用next方法,指针一直往后移动,直到指向最后一个成员
  • d 每调用next方法返回一个包含value和done属性的对象

自定义遍历数据的时候,要想到迭代器

12.生成器函数的声明与调用

生成器函数是ES6提供的一种异步编程解决方案,与传统函数完全不同,就是一个特殊的函数  

  1. // 声明    
  2. function * gen(){// * 可以靠左,也可以靠右,还可以放在中间
  3.   // console.log('hello');
  4.   yield '2020lpl牛批';// yield 语句可以算作函数代码的分隔符
  5.   let two = yield uzi 退役了’;
  6.   console.log(two);
  7.   yield '湖人总冠军';
  8. }
  9. // 执行
  10. let iterator = gen();
  11. // console.log(iterator);// 返回结果是一个迭代器对象
  12. console.log(iterator.next());// 需要执行迭代器对象中的next()方法,才会执行生成器函数
  13. console.log(iterator.next());// 每个next()只会执行里面一个yield语句,这个会输出 ‘uzi 退役了’
  14. // 传递参数 参数将作为上一个yield语句的返回结果
  15. console.log(iterator.next('AAA'));// 第三次调用传递的参数将作为第二个yield 的返回结果 打印为AAA
  16. // 使用for of循环遍历输出
  17. for(let v of gen()){
  18.   console.log(v);// 依次输出yield语句中的值
  19. }

13.Promise 基本使用

Promise 是一个构造函数,用来封装异步操作并可以获取其成功或失败的结果

  • 1、 Promise 构造函数  Promise(excutor){}
  • 2、Promise.prototype.then 有两个参数,两个参数都是函数,

    Promise对象成功状态执行then中第一个函数,失败执行第二个函数

  • 3、 Promise.prototype.catch

    1. // 实例化
    2.     const p = new Promise((resolve, reject) => {
    3.       // 通过resolve,reject这两个函数来改变Promise对象的状态,
    4.       // resolve会改变p的状态为成功,reject会改变p的状态为失败,然后去执行then里面的方法
    5.       // 执行异步操作,以定时器为例,定时器也是异步
    6.       setTimeout(()=>{
    7.         //let data = '异步执行成功';
    8.         // resolve(data);// 调用resolve函数, p会变成一个成功的状态,会执行then中的第一个方法
    9.         let err = '执行失败';
    10.         reject(err);// 调用reject函数,p会变成一个失败的状态,会执行then中的第二个方法
    11.       }, 1000)
    12.     })
    13.     // 成功会调用 promise 对象 then 方法
    14.     p.then(value => {// 成功
    15.       // console.log(value);// 控制台打印:异步执行成功
    16.     }, reason => {// 失败
    17.       console.error(reason)
    18.     })

    Promise.prototype.then 特性   

    1. // then方法的返回结果是Promise对象,对象状态由回调函数的执行结果决定
    2.     const p = new Promise((resolve, reject) => {
    3.       setTimeout(()=>{
    4.         resolve('成功');
    5.         reject('失败');
    6.       }, 1000);
    7.     });
    8.     // then 的返回结果是一个Promise对象,就是result也是一个Promise对象,它的状态由函数的执行结果决定的
    9.     const result = p.then(value => {
    10.       console.log(value);
    11.       // 1.如果返回的结果是 非Promise 类型的属性,状态为成功,返回值return 中的值
    12.       // 如果不写return,函数内部不写return返回结果是undefined,也不是Promise对象,状态也是成功
    13.       // return 123;
    14.       // 2.是 promise 对象, 该对象返回的状态就决定了then方法返回promise对象状态
    15.       return new Promise((resolve, reject)=>{
    16.         // resolve('ok');// then方法返回promise对象状态为成功
    17.         reject('no');// then方法返回promise对象状态为失败
    18.       })
    19.       // 3.抛出错误  then方法返回promise对象状态为失败,错误值为抛出错误的值
    20.       throw new Error('出错了');
    21.     }, reason => {
    22.       console.err(reason);
    23.     });
    24.     console.log(result);
    25.     // 综上总结,then方法可以链式调用  可以改变回调域的现象
    26.     p.then(value=>{}, reason=>{})
    27.       .then(value()=>{}).then();
    28. 复制代码

    举例:多个请求都返回之后,获取其中的数据

    1. const p = new Promise((resolve, reject)=>{
    2.       resolve('第一次返回成功')
    3.     });
    4.     p.then(value=>{
    5.       return new Promise((resolve, reject)=>{
    6.         resolve([value, '第二次返回成功'])
    7.       });
    8.     }).then(value=>{
    9.       return new Promise((resolve, reject)=>{
    10.         resolve([...value, '第三次返回成功'])
    11.       });
    12.     }).then(value=>{
    13.       console.log(value);// 返回值为三次请求都返回成功以后的值
    14.     });

14.集合set

新的数据结构Set(集合),它类似于数组,成员的值都是唯一的,集合实现了iterator接口,所以可以使用扩展运算符和for of遍历
集合的属性和方法

  • 1 size  返回集合的元素个数
  • 2 add  添加一个新元素,返回当前集合
  • 3 delete  删除元素,返回boolean值
  • 4 has  检测集合中是否包含某个元素,返回boolean值
  • 5 clear  清空 ```javascript // 声明     let s = new Set();     let s2 = new Set([1, 2, 3, 6, 7]);     console.log(s2.size);// 5     s2.add(8);// 添加新元素     console.log(s2);// 输出 {1, 2, 3, 6, 7, 8}     s2.delete(8);     console.log(s2);// 输出 {1, 2, 3, 6, 7}     console.log(s2.has(8));// false     // s2.clear();// 清空     let arr = [1, 2, 3, 3, 3, 6, 6, 8];     let arr2 = [1, 3, 6, 7, 8];     // 数组去重     let result = […new Set(arr)];

    // 交集     let result = […new Set(arr)].filter(item => new Set(arr2).has(item));

    // 并集     let result = […new Set([…arr, …arr2])];

    // 差集 arr有arr2中没有     let result = […new Set(arr)].filter(item => !(new Set(arr2).has(item)));

  1. <a name="MXVuG"></a>
  2. # 15.Map集合
  3. 类似于对象,也是键值对的集合,但是 **键 **不限于字符串,各种类型的值(包括对象)都可以当作键,<br />map也实现了 iterator 接口,所以**可以使用扩展运算符和for of进行遍历**
  4. - 1 size  返回 Map 的元素个数
  5. - 2 set  增加一个新元素,返回当前Map
  6. - 3 get  返回键名对象的键值
  7. - 4 has  检测Map中是否包含某个元素,返回boolean值
  8. - 5 clear  清空集合,返回undefined
  9. ```javascript
  10. // 声明
  11.     let m = new Map();
  12.     m.set('name', 'zhangning');
  13.     m.set('change', function(){console.log('变得更努力')});// 键 change 值 一个function
  14.     let key = {company: 'supwisdom'};
  15.     m.set(key, [1, 2, 3]);//键 对象 值 数组
  16.     m.size;// 获取m个数
  17.     m.delete('name');// 删除键值对
  18.     m.get('change');// 获取键对应的值
  19.     // m.clear();// 清空
  20.     for(let v of m){console.log(v);}

16.class 类

通过class可以定义类,新的class写法只是让对象原型的写法更加清晰,更像面向对象编程的语法而已。

  • 1 class  声明类
  • 2 constructor  定义构造函数初始化
  • 3 extends  继承父类
  • 4 super  调用父级构造方法
  • 5 static  定义静态方法和属性
  • 6 父类方法可以重写

es5通过 构造函数实例化 对象的方法

  1. // 人 ,定义一个Perple类
  2.     function People(name, sex) {
  3.       this.name = name;
  4.       this.sex = sex;
  5.     }
  6.     // 这个height这种添加方式是属于函数对象的,不属于实例对象,这样的属性称之为静态成员
  7.     People.height = '180';
  8.     People.prototype.height1 = '100';
  9.     // 添加方法
  10.     People.prototype.play = function(){
  11.       console.log('打篮球');
  12.     }
  13.     let zn = new People('zhangning', '男');
  14.     zn.play();// 输出 打篮球
  15.     console.log(zn);
  16.     console.log(zn.height);// 输出 undefined
  17.     console.log(zn.height1);// 输出 100,必须通过prototype添加才能添加到实例对象上

通过class实现

  1. class People{
  2.       // 静态属性 static,对于static 标注的方法属于类,不属于实例对象
  3.       static height = '100';
  4.       static change(){
  5.         console.log('我可以改变世界');
  6.       }
  7.       // 构造方法 名字不能更改(在使用new People的时候会自动执行实例对象上的constructor方法)       
  8.       constructor(name, sex){
  9.         this.name = name;
  10.         this.sex = sex;
  11.       }
  12.       // 添加方法必须使用该语法,不能使用es5的完整形式(play: function(){} 这种形式不支持,必须使用play()形式)
  13.       // 成员属性
  14.       play(){
  15.         console.log('打篮球');
  16.       }
  17.     }
  18.     let zn = new People('zhangning', '男');
  19.     console.log(zn);
  20.     console.log(zn.height);// undefined  static 标注的方法属于类,不属于实例对象
  21.     console.log(People.height);// 100

使用es5构造函数实现继承 

  1. // 举例 chinese 继承 People 属性
  2.     function People(name, sex) {
  3.       this.name = name;
  4.       this.sex = sex;
  5.     }
  6.     People.prototype.play = function(){
  7.       console.log('打LOL');
  8.     }
  9.     function Student(name, sex, like, height){
  10.       // 通过call方法,改变this值,this指向chinese中的this,也就是chinese的一个实例对象
  11.       People.call(this, name, sex);
  12.       this.like = like;
  13.       this.height = height;
  14.     }
  15.     // 设置子集构造函数原型
  16.     Student.prototype = new People;// 这样就会有父级的一个方法
  17.     Student.prototype.constructor = Student;// 做一个校正,没有这行代码也无所谓
  18.     // 声明子类方法
  19.     Student.prototype.photo = function(){
  20.       console.log('去拍照');
  21.     }
  22.     // 实例化
  23.     const zn = new Student('zhangning', '男', '打篮球', '187');
  24.     console.log(zn)

使用es6 class 类 实现继承 及 父类方法的重写

  1. // 声明父类
  2.     class People{
  3.       // 父类构造方法
  4.       constructor(name, sex) {
  5.         this.name = name;
  6.         this.sex = sex;
  7.       }
  8.       // 父类成员属性
  9.       play(){
  10.         console.log('打LOL');
  11.       }
  12.     }
  13.     // 声明子类 使用extends 继承父类
  14.     class Student extends People {
  15.       // 构造方法
  16.       constructor(name, sex, like, height){
  17.         super(name, sex);// super 就是父类的constructor构造函数,这样调用
  18.         this.like = like;
  19.         this.height = height;
  20.       }
  21.       photo(){
  22.         console.log('去拍照');
  23.       }
  24.       // 对父类中的play方法进行重写,子类是不能去调用父类的同名方法的,
  25.       play(){
  26.         // super(); 不允许,在普通的成员方法里面是不能出现super()去调用父类的同名方法的,会报错,只能完全重写
  27.         console.log('我会打LOL,还会打篮球');
  28.       }
  29.     }
  30.     const zn = new Student('zhangning', '男', '打篮球', '187');
  31.     console.log(zn)
  32. 复制代码

class 中 getter 和 setter 设置

  1. class People{
  2.       get like(){
  3.         return '打篮球';
  4.       }
  5.       set like(newVal){
  6.         // 通过传过来的newVal值,进行操作,改变 like
  7.         console.log('改变like值');
  8.       }
  9.     }
  10.     let p = new People();
  11.     console.log(p.like)// 输出 打篮球
  12.     p.like = 'LOL';// 然后通过 set like 进行操作

17.对象数值扩展

Object.is  

判断两个值是否完全相等

  1. Object.is(1, 1);// true

和 === 很相似,唯一区别就是 NaN === NaN 为 false, Object.is(NaN, NaN) 为true

Object.assign  

对象的合并

  1. const c1 = {name: 'znn'};
  2. const c2 = {name: 'zhangning', height: 187};
  3. Object.assign(c1, c2);// 如果两个对象中存在相同属性,c2 中覆盖c1中的属性内容

Object.setPrototypeOf  

设置原型对象

  1. const zn = {
  2.     name: 'zhangning',
  3. }
  4. const p = {
  5.     h: true,
  6.     sfsfdf: 'fasfasdf'
  7. }
  8. Object.setPrototypeOf(zn, p);// 设置 zn 的原型里面有 p
  9. Object.getPrototypeOf(zn);// 获取 zn 的原型
  10. console.log(zn);// 打印看下

18.数组扩展

Array.from()

Array.from方法用于将两类对象转为真正的数组:类数组的对象( array-like object )和可遍历( iterable )的对象(包括 ES6 新增的数据结构 Set 和Map )。

  1. let arrayLike = {
  2.   '0': 'a',
  3.   '1': 'b',
  4.   '2': 'c',
  5.   length: 3
  6. };
  7. // ES5 的写法
  8. var arr1 = [].slice.call(arrayLike); // ['a', 'b', 'c']
  9. // ES6 的写法
  10. let arr2 = Array.from(arrayLike); // ['a', 'b', 'c']
  11. // NodeList 对象
  12. let ps = document.querySelectorAll('p');
  13. Array.from(ps).forEach(function (p) {
  14.   console.log(p);
  15. });
  16. // arguments 对象
  17. function foo() {
  18. var args = Array.from(arguments);
  19. // ...
  20. }
  21. //字符串转换为字符数组str.split('')
  22. Array.from('hello') // ['h', 'e', 'l', 'l', 'o']
  23. let namesSet = new Set(['a', 'b'])
  24. Array.from(namesSet) // ['a', 'b']
  25. Array.from({ length: 3 }); // [ undefined, undefined, undefined ]

对于还没有部署该方法的浏览器,可以用Array.prototype.slice方法替代:

  1. const toArray = (() =>
  2. Array.from ? Array.from : obj => [].slice.call(obj)
  3. )();

Array.from还可以接受第二个参数,作用类似于数组的map方法,用来对每个元素进行处理,将处理后的值放入返回的数组。

  1. Array.from(arrayLike, x => x * x);
  2. // 等同于
  3. Array.from(arrayLike).map(x => x * x);
  4. Array.from([1, 2, 3], (x) => x * x)
  5. // [1, 4, 9]
  1. //Array.from回调函数
  2. var arr1 = Array.from([1,2,3], function(item){
  3. return item*item;
  4. });
  5. var arr2 = Array.from([1,2,3]).map(function(item){
  6. return item*item;
  7. });
  8. var arr3 = Array.from([1,2,3], (item) => item*item);
  9. console.log(arr1); //[ 1, 4, 9 ]
  10. console.log(arr2); //[ 1, 4, 9 ]
  11. console.log(arr3); //[ 1, 4, 9 ]

值得提醒的是,扩展运算符(…)也可以将某些数据结构转为数组。

  1. // arguments 对象
  2. function foo() {
  3.   var args = [...arguments];
  4. }
  5. // NodeList 对象
  6. [...document.querySelectorAll('div')]

Array.of()

Array.of方法用于将一组值,转换为数组。Array.of总是返回参数值组成的数组。如果没有参数,就返回一个空数组。
Array.of基本上可以用来替代Array()或new Array(),并且不存在由于参数不同而导致的重载。它的行为非常统一。
这个方法的主要目的,是弥补数组构造函数Array()的不足。因为参数个数的不同,会导致Array()的行为有差异

  1. Array() // []
  2. Array(3) // [, , ,]
  3. Array(3, 11, 8) // [3, 11, 8]
  4. Array.of() // []
  5. Array.of(3) // [3]
  6. Array.of(3, 11, 8) // [3,11,8]
  7. Array.of(3).length // 1
  8. Array.of(undefined) // [undefined]
  9. Array.of(1) // [1]
  10. Array.of(1, 2) // [1, 2]

Array.of方法可以用下面的代码模拟实现:

  1. function ArrayOf(){
  2. return [].slice.call(arguments);
  3. }

find() 和 findIndex()

数组实例的find方法,用于找出第一个符合条件的数组成员。它的参数是一个回调函数,所有数组成员依次执行该回调函数,直到找出第一个返回值为true的成员,然后返回该成员。如果没有符合条件的成员,则返回undefined。

  1. [1, 4, -5, 10].find((n) => n < 0)
  2. // -5
  3. [1, 5, 10, 15].find(function(value, index, arr) {
  4. return value > 9;
  5. }) // 10

上面代码中,find方法的回调函数可以接受三个参数,依次为当前的值、当前的位置和原数组。 数组实例的findIndex方法的用法与find方法非常类似,返回第一个符合条件的数组成员的位置,如果所有成员都不符合条件,则返回-1。

  1. [1, 5, 10, 15].findIndex(function(value, index, arr) {
  2. return value > 9;
  3. }) // 2

这两个方法都可以接受第二个参数,用来绑定回调函数的this对象。
另外,这两个方法都可以发现NaN,弥补了数组的IndexOf方法的不足。

  1. [NaN].indexOf(NaN)
  2. // -1
  3. [NaN].findIndex(y => Object.is(NaN, y))
  4. // 0

fill()

fill()方法使用给定值,填充一个数组。

  1. ['a', 'b', 'c'].fill(7)
  2. // [7, 7, 7]
  3. new Array(3).fill(7)
  4. // [7, 7, 7]
  5. ['a', 'b', 'c'].fill(7, 1, 2)
  6. // ['a', 7, 'c']

上面代码表明,fill方法用于空数组的初始化非常方便。数组中已有的元素,会被全部抹去。
fill()方法还可以接受第二个和第三个参数,用于指定填充的起始位置和结束位置

  1. ['a', 'b', 'c'].fill(7, 1, 2)
  2. // ['a', 7, 'c']

entries() , keys() 和 values()

ES6 提供三个新的方法 —— entries(),keys()和values() —— 用于遍历数组。它们都返回一个遍历器对象,可以用for…of循环进行遍历,唯一的区别是keys()是对键名的遍历、values()是对键值的遍历,entries()是对键值对的遍历。

  1. for (let index of ['a', 'b'].keys()) {
  2. console.log(index);
  3. }
  4. // 0
  5. // 1
  6. for (let elem of ['a', 'b'].values()) {
  7. console.log(elem);
  8. }
  9. // 'a'
  10. // 'b'
  11. for (let [index, elem] of ['a', 'b'].entries()) {
  12. console.log(index, elem);
  13. }
  14. // 0 "a"
  15. // 1 "b"

如果不使用for…of循环,可以手动调用遍历器对象的next方法,进行遍历。

  1. let letter = ['a', 'b', 'c'];
  2. let entries = letter.entries();
  3. console.log(entries.next().value); // [0, 'a']
  4. console.log(entries.next().value); // [1, 'b']
  5. console.log(entries.next().value); // [2, 'c']

includes()

ES5中,我们常用数组的indexOf方法,检查是否包含某个值。indexOf方法有两个缺点,一是不够语义化,它的含义是找到参数值的第一个出现位置,所以要去比较是否不等于 -1 ,表达起来不够直观。二是,它内部使用严格相当运算符( === )进行判断,这会导致对NaN的误判。

  1. [NaN].indexOf(NaN)
  2. // -1
  3. includes使用的是不一样的判断算法,就没有这个问题。
  4. [NaN].includes(NaN)
  5. // true

Array.prototype.includes方法返回一个布尔值,表示某个数组是否包含给定的值,与字符串的includes方法类似。该方法属于 ES7 ,但 Babel 转码器已经支持。

  1. [1, 2, 3].includes(2); // true
  2. [1, 2, 3].includes(4); // false
  3. [1, 2, NaN].includes(NaN); // true

该方法的第二个参数表示搜索的起始位置,默认为 0 。如果第二个参数为负数,则表示倒数的位置,如果这时它大于数组长度(比如第二个参数为 -4 ,但数组长度为 3 ),则会重置为从 0 开始。

  1. [1, 2, 3].includes(3, 3); // false
  2. [1, 2, 3].includes(3, -1); // true

下面代码用来检查当前环境是否支持该方法,如果不支持,部署一个简易的替代版本。

  1. const contains = (() =>
  2. Array.prototype.includes
  3. ? (arr, value) => arr.includes(value)
  4. : (arr, value) => arr.some(el => el === value)
  5. )();
  6. contains(["foo", "bar"], "baz"); // => false

另外, Map 和 Set 数据结构有一个has方法,需要注意与includes区分。
Map 结构的has方法,是用来查找键名的,比如Map.prototype.has(key)、WeakMap.prototype.has(key)、Reflect.has(target, propertyKey)。
Set 结构的has方法,是用来查找值的,比如Set.prototype.has(value)、WeakSet.prototype.has(value)。

数组的空位

数组的空位指,数组的某一个位置没有任何值。比如,Array构造函数返回的数组都是空位。
注意,空位不是undefined,一个位置的值等于undefined,依然是有值的。空位是没有任何值,in运算符可以说明这一点。

  1. 0 in [undefined, undefined, undefined] // true
  2. 0 in [, , ,] // false

上面代码说明,第一个数组的 0 号位置是有值的,第二个数组的 0 号位置没有值。
ES5 对空位的处理,已经很不一致了,大多数情况下会忽略空位。

  • forEach() , filter() , every() 和some()都会跳过空位。
  • map()会跳过空位,但会保留这个值
  • join()和toString()会将空位视为undefined,而undefined和null会被处理成空字符串。
    1. // forEach方法
    2. [,'a'].forEach((x,i) => console.log(i)); // 1
    3. // filter方法
    4. ['a',,'b'].filter(x => true) // ['a','b']
    5. // every方法
    6. [,'a'].every(x => x==='a') // true
    7. // some方法
    8. [,'a'].some(x => x !== 'a') // false
    9. // map方法
    10. [,'a'].map(x => 1) // [,1]
    11. // join方法
    12. [,'a',undefined,null].join('#') // "#a##"
    13. // toString方法
    14. [,'a',undefined,null].toString() // ",a,,"
    ES6则是明确将空位转为undefined。
    1. //Array.from方法会将数组的空位,转为undefined,也就是说,这个方法不会忽略空位。
    2. Array.from(['a',,'b']) // [ "a", undefined, "b" ]
    3. //扩展运算符(...)也会将空位转为undefined。
    4. [...['a',,'b']] // [ "a", undefined, "b" ]
    5. //copyWithin()会连空位一起拷贝。
    6. [,'a','b',,].copyWithin(2,0) // [,"a",,"a"]
    7. //fill()会将空位视为正常的数组位置。
    8. new Array(3).fill('a') // ["a","a","a"]
    9. //for...of循环也会遍历空位。
    10. let arr = [, ,];
    11. for (let i of arr) {
    12. console.log(1);
    13. }
    14. // 1
    15. // 1
    16. //上面代码中,数组arr有两个空位,for...of并没有忽略它们。如果改成map方法遍历,空位是会跳过的。
    17. //entries()、keys()、values()、find()和findIndex()会将空位处理成undefined。
    18. // entries()
    19. [...[,'a'].entries()] // [[0,undefined], [1,"a"]]
    20. // keys()
    21. [...[,'a'].keys()] // [0,1]
    22. // values()
    23. [...[,'a'].values()] // [undefined,"a"]
    24. // find()
    25. [,'a'].find(x => true) // undefined
    26. // findIndex()
    27. [,'a'].findIndex(x => true) // 0
    28. //由于空位的处理规则非常不统一,所以建议避免出现空位。