Skip to content

ES6~ES13-类和对象

ES6 是一个重要的里程碑,它增加了许多新的语言特性、模块系统、类等等,进一步增强了 JS 的表现力和可维护性。
随着 ES6 的发布,JS 的发展进入了一个快速的时期,每年都会发布一些新的语言特性,从 ES7 到 ES13,新增了大量的新特性和语言功能,例如异步函数、可选链、空值合并操作符、动态导入等等。
从本章开始,我们将介绍从 ES6 到 ES13 的各种语言特性和功能,并给出一些实际应用场景和代码示例。

一. ES6 定义类

1.1. 认识 class 定义类

我们会发现,按照前面的构造函数形式创建 ,不仅仅和编写普通的函数过于相似,而且代码并不容易理解。

  • 在 ES6(ECMAScript2015)新的标准中使用了 class 关键字来直接定义类;
  • 但是类本质上依然是前面所讲的构造函数、原型链的语法糖而已;
  • 所以学好了前面的构造函数、原型链更有利于我们理解类的概念和继承关系;

那么,如何使用 class 来定义一个类呢?

  • 可以使用两种方式来声明类:类声明和类表达式;
javascript
// 类的声明
class Person {}

// 类的表达式
var Student = class {};

接着我们就可以使用 new 操作符调用类:

javascript
var p1 = new Person();
var p2 = new Person();
console.log(p1, p2);

我们来研究一下类的一些特性:

  • 你会发现它和我们的构造函数的特性其实是一致的;
javascript
var p = new Person();

console.log(Person); // [class Person]
console.log(Person.prototype); // {}
console.log(Person.prototype.constructor); // [class Person]

console.log(p.__proto__ === Person.prototype); // true

console.log(typeof Person); // function

1.2. 类的构造函数

如果我们希望在创建对象的时候给类传递一些参数,这个时候应该如何做呢?

  • 每个类都可以有一个自己的构造函数(方法),这个方法的名称是固定的 constructor;
  • 当我们通过 new 操作符,操作一个类的时候会调用这个类的构造函数 constructor;
  • 每个类只能有一个构造函数,如果包含多个构造函数,那么会抛出异常;
javascript
class Person {
  constructor(name, age, height) {
    this.name = name;
    this.age = age;
    this.height = height;
  }
}

var p1 = new Person("why", 18, 1.88);
console.log(p1);

当我们通过 new 关键字操作类的时候,会调用这个 constructor 函数,并且执行如下操作:

  • 1.在内存中创建一个新的对象(空对象);
  • 2.这个对象内部的[[prototype]]属性会被赋值为该类的 prototype 属性;
  • 3.构造函数内部的 this,会指向创建出来的新对象;
  • 4.执行构造函数的内部代码(函数体代码);
  • 5.如果构造函数没有返回非空对象,则返回创建出来的新对象;

1.3. 类的方法定义


1.3.1. 实例方法

在上面我们定义的属性都是直接放到了 this 上,也就意味着它是放到了创建出来的新对象中:

  • 在前面我们说过对于实例的方法,我们是希望放到原型上的,这样可以被多个实例来共享;
  • 这个时候我们可以直接在类中定义;
javascript
class Person {
  constructor(name, age, height) {
    this.name = name;
    this.age = age;
    this.height = height;
  }

  running() {
    console.log(this.name + " running~");
  }

  eating() {
    console.log(this.name + " eating~");
  }
}

var p1 = new Person("why", 18, 1.88);
console.log(p1);

p1.running();
p1.eating();
// [ 'constructor', 'running', 'eating' ]
console.log(Object.getOwnPropertyNames(Person.prototype));

我们也可以查看它们的属性描述符:

  • 会发现它们的 enumerable 都是为 false 的;
javascript
console.log(Object.getOwnPropertyDescriptors(Person.prototype));

1.3.2. 访问器方法

我们之前讲对象的属性描述符时有讲过对象可以添加 setter 和 getter 函数的,那么类也是可以的:

javascript
class Person {
  constructor(name) {
    this._name = name;
  }

  set name(newName) {
    console.log("调用了name的setter方法");
    this._name = newName;
  }

  get name() {
    console.log("调用了name的getter方法");
    return this._name;
  }
}

var p = new Person("why");
console.log(p.name);
p.name = "kobe";
console.log(p.name);

但是和直接在对象中定义不同的是,类中的 setter 和 getter 方法是放到原型上的:

javascript
var p = new Person("why");
console.log(p.name);
p.name = "kobe";
console.log(p.name);

var obj = {
  _name: "",
  set name(newName) {
    this._name = newName;
  },
  get name() {
    return this._name;
  },
};

console.log(obj);
console.log(Object.getOwnPropertyDescriptors(p.__proto__));

1.3.3. 静态方法

静态方法通常用于定义直接使用类来执行的方法,不需要有类的实例,使用 static 关键字来定义:

javascript
class Person {
  constructor(age) {
    this.age = age;
  }

  static create() {
    return new Person(Math.floor(Math.random() * 100));
  }
}

for (var i = 0; i < 10; i++) {
  console.log(Person.create());
}

二. ES6 类的继承

2.1. extends 关键字

前面我们花了很大的篇幅讨论了在 ES5 中实现继承的方案,虽然最终实现了相对满意的继承机制,但是过程却依然是非常繁琐的。

在 ES6 中新增了使用 extends 关键字,可以方便的帮助我们实现继承:

javascript
class Person {}

class Student extends Person {}

我们知道继承可以让我们复用父类的一些代码结构,比如继承属性和方法:

javascript
class Person {
  constructor(name, age) {
    this.name = name;
    this.age = age;
  }

  running() {
    console.log(this.name + " running~");
  }

  eating() {
    console.log(this.name + " eating~");
  }
}

class Student extends Person {
  constructor(name, age, sno) {
    super(name, age);
    this.sno = sno;
  }

  studying() {
    console.log(this.name + " studying~");
  }
}

var stu = new Student("why", 18, 111);

2.2. super 关键字

我们会发现在上面的代码中我使用了一个 super 关键字,这个 super 关键字有不同的使用方式:

  • 注意:在子(派生)类的构造函数中使用 this 或者返回默认对象之前,必须先通过 super 调用父类的构造函数!
  • super 的使用位置有三个:子类的构造函数、实例方法、静态方法;
javascript
// 调用 父对象/父类 的构造函数
super([arguments]);

// 调用 父对象/父类 上的方法
super.functionOnParent([arguments]);

下面的代码会报错,因为我们没有调用 super:

javascript
class Person {}

class Student extends Person {
  constructor(sno) {
    // ReferenceError: Must call super constructor in derived class before accessing 'this' or returning from derived constructor
  }
}

var stu = new Student();

我们可以在子类的方法中调用父类的方法:

javascript
class Person {
  eating() {
    console.log(this.name + " eating~");
  }

  static create() {
    console.log("Person create");
  }
}

class Student extends Person {
  // 方法的重写
  eating() {
    console.log("做完作业~");
    super.eating();
  }

  static create() {
    super.create();
    console.log("Student create");
  }
}

var stu = new Student();
stu.eating();

Student.create();

2.3. 继承内置类

我们也可以让我们的类继承自内置类,比如 Array:

javascript
class HYArray extends Array {
  lastItem() {
    return this[this.length - 1];
  }
}

var array = new HYArray(10, 20, 30);
console.log(array.lastItem());

array.filter((item) => {
  console.log(item);
});

调用 Array 的方法返回 Array 类型:

javascript
class HYArray extends Array {
  lastItem() {
    return this[this.length - 1];
  }

  static get [Symbol.species]() {
    return Array;
  }
}

var array = new HYArray(10, 20, 30);
console.log(array.lastItem());

var newArr = array.filter((item) => {
  console.log(item);
});

console.log(newArr instanceof HYArray); // false
console.log(newArr instanceof Array); // true

三. ES 对象的增强

ES6 中对 对象字面量 进行了增强,称之为 Enhanced object literals(增强对象字面量)。

3.1. 属性的简写

在开发中,对象中的属性可能会经常来自变量,并且变量的名称和属性的名称是相同的,这个时候我们可以使用简写:

  • 英文称之为 Property Shorthand
javascript
var name = "why";
var age = 18;

// ES5的写法
var obj1 = {
  name: name,
  age: age,
};

// ES6的增强写法
var obj2 = {
  name,
  age,
};

3.2. 方法的简写

另外一个对象增强的写法是针对方法的:

  • 英文称之为 Method Shorthand
javascript
// ES5的方法写法
var info1 = {
  foo: function () {},
  bar: function () {},
};

// ES6的增强写法
var info2 = {
  foo() {},
  bar() {},
};

3.3. 计算属性名

在 ES5 中,如果一个对象中属性的名称来自一个变量,或者需要其他的方法计算得到,那么我们需要这样来做:

javascript
var name = "three";
var obj = {
  one: 1,
  two: 2,
};

obj[name] = 3;

在 ES6 中,我们可以直接在字面量中编写计算属性名:

英文称之为 Computed Property Names

javascript
var name = "three";
var obj = {
  one: 1,
  two: 2,
  [name]: 3,
};

console.log(obj);

四. 解构 Destructuring

ES6 中新增了一个从数组或对象中方便获取数据的方法,称之为解构 Destructuring。

4.1. 数组的解构

数组的解构就是从数组中获取我们需要的数组:

javascript
var names = ["abc", "cba", "nba"];

var [name1, name2, name3] = names;
console.log(name1, name2, name3);

数组的解构必须按照数据的顺序依次获取,如果我们只想获取第二个和第三个,那么可以有如下语法:

javascript
var [, nameb, namec] = names;
console.log(nameb, namec);

如果我们希望解构出来一个元素,其他元素继续放到另外一个数组中:

javascript
var [namea, ...newNames] = names;
console.log(namea, newNames);

如果我们解构的数据数量大于数组中原本的数据数量,那么会返回 undefined:

javascript
var [namex, namey, namez, namem] = names;
console.log(namem); // undefined

我们可以在解构出来的数据为 undefined 的时候,给它一个默认值:

javascript
var [namex, namey, namez, namem = "aaa"] = names;
console.log(namem); // aaa

4.2. 对象的解构

对象的解构和数组的解构是相似的,不同之处在于:

  • 数组中的元素是按照顺序排列的,并且我们只能根据顺序来确定需要获取的数据;
  • 对象中的数据由 key 和 value 组成,我们可以通过 key 来获取想要的 value;
javascript
var obj = {
  name: "why",
  age: 18,
  height: 1.88,
};

var { name, age, height } = obj;
console.log(name, age, height);

因为对象是可以通过 key 来解构的,所以它对顺序、个数都没有要求:

javascript
var { height, name, age } = obj;
console.log(name, age, height);

var { age, height } = obj;
console.log(age, height);

如果我们对变量的名称不是很满意,那么我们可以重新命名:

javascript
var { name: whyName, age: whyAge } = obj;
console.log(whyName, whyAge);

我们也可以给变量一个默认值:

javascript
var { name, address = "广州市" } = obj;
console.log(name, address);

4.3. 解构的使用

解构目前在开发中使用是非常多的:

  • 比如在开发中拿到一个变量时,自动对其进行解构使用;
  • 比如对函数的参数进行解构;
image-20210927170135546image-20210927170322048