ES6新增了class来替代之前的构造函数,并且通过extends关键字可以轻易实现继承。不过ES的概念中,暂时还没有class这一类型,不管从哪方面来看,class都是对之前的继承方案的封装,其本质上是函数(Function的实例) 。了解一下ES6之前的各种继承方案,有助于加深对class继承的理解。
一、创建对象
1. 工厂模式
当需要创建多个对象实例,且他们的属性高度重复时,无论是通过对象字面量来创建,还是使用new Object来创建,都非常麻烦。采用工厂模式,可以很方便地批量创建多个具有相同属性的对象实例,为此,需要定义一个工厂函数:
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 
 | function createPerson(name, age, gender, ...rest) {
 
 const person = {};
 
 person.name = name;
 person.age = age;
 person.gender = gender;
 
 
 return person;
 }
 
 
 
 
 const cc = createPerson("cc", 18, "男");
 
 const yy = createPerson("yy", 18);
 
 
 
 | 
可见,工厂函数就是一个普通函数,创建空对象、增强对象、返回对象 三步走,它可以解决批量创建相似对象的问题,但是创建出来的对象没有标识,我们难以区分他们是什么类型。
2. 构造函数模式
构造函数模式在工厂模式的基础上加以改进。无需显示地创建空对象,且把属性赋值给this,也不需要显示return,使用new操作符来创建实例对象。
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 
 | function Person(name, age, gender, ...rest) {
 
 
 
 this.name = name;
 this.age = age;
 this.gender = gender;
 this.playGame = (gameName) => console.log(`${gameName} start!!`);
 
 
 }
 
 
 const cc = new Person("cc", 18, "男");
 
 const yy = new Person("yy", 18);
 
 
 
 console.log(cc.constructor === Person);
 
 console.log(cc instanceof Person);
 console.log(yy instanceof Person);
 
 | 
构造函数模式不仅代码上看更加简洁(不需要显示创建对象以及return),而且通过new操作符来创建的实例对象,具有对象标识,可以轻松地使用instanceof操作符来检测它们是否属于某一类型。
此外,构造函数也是函数,除了与new操作符搭配使用以外,也可以当作普通函数来直接调用。此时,如果没有使用call/apply等方式来改变this指向,则this会指向Global对象,在浏览器中即window对象。
| 12
 3
 4
 5
 6
 7
 8
 9
 
 | Person("ww", 20);
 
 console.log(name, age);
 
 
 const boy = {};
 Person.call(boy, "cc", 18);
 console.log(boy);
 
 | 
对象函数模式的问题,在于其定义的方法会在每个实例上都创建一边。如上栗子中,cc和yy都有playGame()方法,但是他们的方法并不是引用的同一个,而是各自单独的实例,即cc.playGame === yy.playGame会得到false。这显然会造成不必要的浪费。我们可以把函数定义转移到构造函数外部,来解决这个问题。
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 
 | function Person(name, age, gender, ...rest) {
 
 this.name = name;
 this.age = age;
 this.gender = gender;
 
 
 this.playGame = playGame;
 }
 
 function playGame(gameName) {
 console.log(`${gameName} start!!`);
 }
 
 const cc = new Person("cc"),
 yy = new Person("yy");
 
 cc.playGame === yy.playGame;
 
 | 
如此这般,虽然可以解决实例对象共享方法的问题,但是由于方法定义在构造函数外部,导致全局可调用该函数,而且一旦共享的方法多了,就需要在外部定义很多函数,不方便管理与维护。
3. 原型模式
关于原型在此不过多赘述,每个函数都会创建一个prototype属性,即函数的原型对象,包含其实例对象所共享的方法和属性。因此,在构造函数中把值赋给原型对象,则可以让其实例对象共享这些值/方法。这里主要有两种方式,一种是给已有的原型对象添加新的属性和方法:
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 
 | function Person() {
 
 const prototype = Person.prototype;
 
 prototype.name = "cc";
 prototype.age = 18;
 prototype.playGame = (game) => console.log(`${game} start!!`);
 }
 
 const cc = new Person(),
 yy = new Person();
 
 console.log(cc.name);
 console.log(yy.name);
 
 | 
另一种是把需要添加的属性/方法集中在一个对象中,然后赋给原型对象。这会导致构造函数的原型被重写,与之前已有的原型不再有关联,也得手动让constructor属性重新指向构造函数本身。
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 
 | function Person() {
 
 const prototype = {
 name: "cc",
 age: 18,
 playGame(game) {
 console.log(`${game} start!!`);
 },
 
 constructor: Person,
 };
 
 
 Person.prototype = prototype;
 }
 
 | 
由于重写原型会使构造函数的原型指向一个新的对象,这会导致在执行重写原型的操作前后实例化的对象具有不同的原型,这点尤为值得注意。
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 
 | function Person() {
 }
 
 Person.prototype.name = "cc";
 Person.prototype.playGame = (game) => console.log(`${game} start!!`);
 
 
 const cc = new Person();
 
 
 Person.prototype = {
 name: "yy",
 playGame(game) {
 console.log("You are not allowed to play game !!");
 },
 constructor: Person,
 };
 
 const yy = new Person();
 
 console.log(cc.name);
 console.log(yy.name);
 
 cc.playGame("Don't Starve Together");
 yy.playGame("Don't Starve Together");
 
 | 
原型模式弱化了向构造函数传参来自定义属性值的能力,且通过原型共享的引用类型也会在各个实例之间相互影响,因此,原型模式基本不会单独应用。往往是将构造函数模式和原型模式进行结合。
二、继承
1. 原型链继承
原型链的概念在此不做赘述。将一个构造函数 A 的原型,重写为另一个构造函数 B 的一个实例对象,由于该实例对象可以访问构造函数 B 的原型上的属性和方法,当成为构造函数 A 的原型时,则构造函数 A 的实例对象也可以访问构造函数 B 的原型上的属性/方法。
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 
 | function B() {this.age = 18;
 B.prototype.name = "b";
 b.prototype.playGame = function (game) {
 console.log(`${game} start!!`);
 };
 }
 
 
 function A() {
 
 const prototype = new B();
 A.prototype = prototype;
 }
 
 
 const a = new A();
 a.name;
 a.playGame("Don't Starve Together");
 
 
 a.age;
 
 | 
通过原型链继承,弱化了向构造函数传参的能力,且父类构造函数的实例属性/方法也会成为子类构造函数的原型属性/方法,这在某些时候会导致问题。
2. 经典继承:盗用构造函数
通过原型链继承,父类的实例属性/方法会变成子类的原型属性/方法,且难以通过向构造函数传参来自定义属性值,这显然不是我们想要的。通过call/apply在子类构造函数中来盗用父类构造函数,可以让父类构造函数的实例属性/方法在子类中也同样赋值操作一遍,子类即可获得父类的实例属性/方法,但是并没有继承父类的原型属性/方法。
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 
 | function B(name, age) {this.name = name;
 this.age = age;
 B.prototype.playGame = function (game) {
 console.log(`${game} start!!`);
 };
 }
 
 
 
 function A(name, age) {
 B.call(this, name, age);
 }
 
 const a = new A("a", 5);
 const b = new B("b", 10);
 a.name;
 a.age;
 b.name;
 b.age;
 
 b.playGame("Don't Starve Together");
 
 
 
 a.playGame("Don't Starve Together");
 
 | 
盗用构造函数的方式无法继承原型上的内容,这一点可以通过原型链继承来弥补。因此将二者组合,便可以实现一个完整的继承。
3. 组合继承
组合继承是将原型链继承和盗用构造函数继承结合起来。
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 
 | function Person(name) {
 this.name = name;
 this.age = 20;
 Person.prototype.playGame = function (game) {
 console.log(`${game} start!!`);
 };
 }
 
 
 function Student(name, age, school) {
 
 Person.call(this, name, age);
 
 this.school = school;
 }
 
 
 const prototype = new Person();
 prototype.constructor = Student;
 Student.prototype = prototype;
 
 const cc = new Student("cc", 18, "cc-school");
 
 
 cc.name;
 cc.age;
 cc.school;
 
 
 cc.playGame("Don't Starve Together");
 
 
 
 
 
 cc.__proto__.name;
 
 cc.__proto__.age;
 
 
 
 cc.name;
 cc.age;
 
 | 
上面的栗子中,可以看到,在继承的实现过程中,父类构造函数调用了两次:一次是盗用父类构造函数,通过call调用,另一次是原型链继承时,通过new操作符调用。父类构造函数的实例属性/方法,既通过盗用构造函数继承为子类的实例属性/方法,又通过原型链继承成为子类的原型属性/方法。这在实现了完整的继承的同时,也造成了不必要的浪费。
4. 寄生式继承
组合继承的缺陷主要来自其中的原型链继承,会把父类的实例属性/对象也变为子类的原型属性/对象,而我们只希望在原型链中继承父类的原型链,而不希望父类的实例属性/方法也成为子类的原型链的一部分。寄生式继承为解决这个问题提供了思路。
寄生式继承主要分三步:创建对象、增强对象、返回对象。主要用于无需创建构造函数,而是基于一个对象obj,创建另一个增强版的对象obj2,来实现对obj的继承。
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 
 | function create(obj) {
 const newObj = Object.create(obj);
 
 newObj.name = "new-obj";
 
 return newObj;
 }
 
 const person = {
 name: "person",
 age: 18,
 playGame(game) {
 console.log(`${game} start!!`);
 },
 };
 
 const p2 = create(person);
 p2.name;
 p2.age;
 p2.playGame("Don't Starve Together");
 
 | 
这里使用了Object.create()方法,在只传一个对象作为参数时,这个方法会返回一个以该对象为原型的空对象。
| 12
 3
 4
 5
 
 | function obj_create(obj) {function Fun() {}
 Fun.prototype = obj;
 return new F();
 }
 
 | 
这样就能避免组合继承中通过原型链继承带来的问题,且父类构造函数不再需要调用两次。
5. 最佳模式:寄生式组合继承
将寄生式继承的思路引入到组合继承中,成为ES5引用类型的最佳继承模式。
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 
 | function newPrototype(SuperType, SubType){
 
 const prototype = Object.create(SuperType.prototype)
 
 prototype.constructor = SubType
 SubType.prototype = prototype
 }
 
 
 function Person(name, age){
 this.name = name
 this.age = age
 Person.prototype.playGame = (game) => console.log(`${game} start!!`)
 }
 
 
 function Student(name, age, school){
 
 Person.call(this, name, age)
 
 this.school = school
 }
 
 newPrototype(Person, Student)
 
 const cc = new Student('cc', 18, 'cc-school')
 cc.name
 cc.age
 cc.school
 
 cc..playGame("Don't Starve Together")
 
 | 
ES5及其之前的对象创建与继承基本都回顾了一遍,有空再整理下ES6的class及其与构造函数的对比。