ES6
新增了class
来替代之前的构造函数,并且通过extends
关键字可以轻易实现继承。不过ES
的概念中,暂时还没有class
这一类型,不管从哪方面来看,class
都是对之前的继承方案的封装,其本质上是函数(Function
的实例) 。了解一下ES6
之前的各种继承方案,有助于加深对class
继承的理解。
一、创建对象
1. 工厂模式
当需要创建多个对象实例,且他们的属性高度重复时,无论是通过对象字面量来创建,还是使用new Object
来创建,都非常麻烦。采用工厂模式,可以很方便地批量创建多个具有相同属性的对象实例,为此,需要定义一个工厂函数:
1 2 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
操作符来创建实例对象。
1 2 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
对象。
1 2 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
。这显然会造成不必要的浪费。我们可以把函数定义转移到构造函数外部,来解决这个问题。
1 2 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
属性,即函数的原型对象,包含其实例对象所共享的方法和属性。因此,在构造函数中把值赋给原型对象,则可以让其实例对象共享这些值/方法。这里主要有两种方式,一种是给已有的原型对象添加新的属性和方法:
1 2 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
属性重新指向构造函数本身。
1 2 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; }
|
由于重写原型会使构造函数的原型指向一个新的对象,这会导致在执行重写原型的操作前后实例化的对象具有不同的原型,这点尤为值得注意。
1 2 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 的原型上的属性/方法。
1 2 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
在子类构造函数中来盗用父类构造函数,可以让父类构造函数的实例属性/方法在子类中也同样赋值操作一遍,子类即可获得父类的实例属性/方法,但是并没有继承父类的原型属性/方法。
1 2 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. 组合继承
组合继承是将原型链继承和盗用构造函数继承结合起来。
1 2 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
的继承。
1 2 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()
方法,在只传一个对象作为参数时,这个方法会返回一个以该对象为原型的空对象。
1 2 3 4 5
| function obj_create(obj) { function Fun() {} Fun.prototype = obj; return new F(); }
|
这样就能避免组合继承中通过原型链继承带来的问题,且父类构造函数不再需要调用两次。
5. 最佳模式:寄生式组合继承
将寄生式继承的思路引入到组合继承中,成为ES5
引用类型的最佳继承模式。
1 2 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
及其与构造函数的对比。