文以《JavaScript高级程序设计》上的内容为骨架,补充了ES6 Class的相关内容,从我认为更容易理解的角度将继承这件事叙述出来,希望大家能有所收获。

1. 继承分类


先来个整体印象。如图所示,JS中继承可以按照是否使用object函数(在下文中会提到),将继承分成两部分(Object.create是ES5新增的方法,用来规范化这个函数)。

其中,原型链继承和原型式继承有一样的优缺点,构造函数继承与寄生式继承也相互对应。寄生组合继承基于Object.create, 同时优化了组合继承,成为了完美的继承方式。ES6 Class Extends的结果与寄生组合继承基本一致,但是实现方案又略有不同。

下面马上进入正题:

2. 继承方式


2.1 原型式继承

核心:将父类的实例作为子类的原型。

SubType.prototype = new SuperType()
// 所有涉及到原型链继承的继承方式都要修改子类构造函数的指向,
// 否则子类实例的构造函数会指向SuperType。
SubType.prototype.constructor = SubType;

优点:父类方法可以复用。

缺点:

1、父类的引用属性会被所有子类实例共享

2、子类构建实例时不能向父类传递参数

2.2 构造函数继承

核心:将父类构造函数的内容复制给了子类的构造函数。这是所有继承中唯一一个不涉及到prototype的继承。

SuperType.call(SubType);

优点:和原型链继承完全反过来

1、父类的引用属性不会被共享

2、子类构建实例时可以向父类传递参数

缺点:父类的方法不能复用,子类实例的方法每次都是单独创建的。

2.3 组合继承

核心:原型式继承和构造函数继承的组合,兼具了二者的优点。

function SuperType() {
this.name = 'parent';
this.arr = [1, 2, 3];
} SuperType.prototype.say = function() {
console.log('this is parent')
} function SubType() {
SuperType.call(this) // 第二次调用SuperType
} SubType.prototype = new SuperType() // 第一次调用SuperType

优点:

1、父类的方法可以被复用

2、父类的引用属性不会被共享

3、子类构建实例时可以向父类传递参数

缺点:调用了两次父类的构造函数,第一次给子类的原型添加了父类的name, arr属性,第二次又给子类的构造函数添加了父类的name, arr属性,从而覆盖了子类原型中的同名参数。这种被覆盖的情况造成了性能上的浪费。

2.4 原型式继承

核心:原型式继承的object方法本质上是对参数对象的一个浅复制。

优点:父类方法可以复用。

缺点:

1、父类的引用属性会被所有子类实例共享

2、子类构建实例时不能向父类传递参数

function object(o){
function F(){}
F.prototype = o;
return new F();
} var person = {
name: "Nicholas",
friends: ["Shelby", "Court", "Van"]
}; var anotherPerson = object(person);
anotherPerson.name = "Greg";
anotherPerson.friends.push("Rob"); var yetAnotherPerson = object(person);
yetAnotherPerson.name = "Linda";
yetAnotherPerson.friends.push("Barbie");
alert(person.friends); //"Shelby,Court,Van,Rob,Barbie"

ECMAScript 5 通过新增 Object.create()方法规范化了原型式继承。这个方法接收两个参数:一 个用作新对象原型的对象和(可选的)一个为新对象定义额外属性的对象。在传入一个参数的情况下, Object.create()与 object()方法的行为相同。——《JAVASCript高级编程》

所以上文中代码可以转变为:

var yetAnotherPerson = object(person); =>
var yetAnotherPerson = Object.create(person);

2.5 寄生式继承

核心:使用原型式继承获得一个目标对象的浅复制,然后增强这个浅复制的能力。

优缺点:仅提供一种思路,没什么优点。

function createAnother(original){
var clone=object(original); //通过调用函数创建一个新对象
clone.sayHi = function(){ //以某种方式来增强这个对象
alert("hi");
};
return clone; //返回这个对象
} var person = {
name: "Nicholas",
friends: ["Shelby", "Court", "Van"]
}; var anotherPerson = createAnother(person);
anotherPerson.sayHi(); //"hi"

2.6 寄生组合继承

刚才说到组合继承有一个会两次调用父类的构造函数造成浪费的缺点,寄生组合继承就可以解决这个问题。

function inheritPrototype(subType, superType){
var prototype = object(superType.prototype); // 创建了父类原型的浅复制
prototype.constructor = subType; // 修正原型的构造函数
subType.prototype = prototype; // 将子类的原型替换为这个原型
} function SuperType(name){
this.name = name;
this.colors = ["red", "blue", "green"];
} SuperType.prototype.sayName = function(){
alert(this.name);
}; function SubType(name, age){
SuperType.call(this, name);
this.age = age;
}
// 核心:因为是对父类原型的复制,所以不包含父类的构造函数,
// 也就不会调用两次父类的构造函数造成浪费
inheritPrototype(SubType, SuperType);
SubType.prototype.sayAge = function(){
alert(this.age);
}

  

优缺点:这是一种完美的继承方式。

2.7 ES6 Class extends

核心: ES6继承的结果和寄生组合继承相似,本质上,ES6继承是一种语法糖。但是,寄生组合继承是先创建子类实例this对象,然后再对其增强;而ES6先将父类实例对象的属性和方法,加到this上面(所以必须先调用super方法),然后再用子类的构造函数修改this。

class A {}

class B extends A {
constructor() {
super();
}
}

ES6实现继承的具体原理:

class A {
} class B {
} Object.setPrototypeOf = function (obj, proto) {
obj.__proto__ = proto;
return obj;
} // B 的实例继承 A 的实例
Object.setPrototypeOf(B.prototype, A.prototype); // B 继承 A 的静态属性
Object.setPrototypeOf(B, A);

ES6继承与ES5继承的异同:

相同点:本质上ES6继承是ES5继承的语法糖。

不同点:

1、ES6继承中子类的构造函数的原型链指向父类的构造函数,ES5中使用的是构造函数复制,没有原型链指向。

2、ES6子类实例的构建,基于父类实例,ES5中不是。

3. 总结


  • 1、ES6 Class extends是ES5继承的语法糖
  • 2、JS的继承除了构造函数继承之外都基于原型链构建的
  • 3、可以用寄生组合继承实现ES6 Class extends,但是还是会有细微的差别

参考文章:

  • 1、《js继承、构造函数继承、原型链继承、组合继承、组合继承优化、寄生组合继承》
  • 2、《JavaScript高级编程》

 

最新文章

  1. SQL Server 中存储过程的练习
  2. iOS开发 差间距滚动
  3. 当调用List Remove 失效时 [C#] .
  4. javascript 判断整数
  5. Genymotion如何访问本地服务器?
  6. deeplearning.ai 卷积神经网络 Week 3 目标检测 听课笔记
  7. redis结合自定义注解实现基于方法的注解缓存,及托底缓存的实现
  8. Python-定时爬取指定城市天气(一)-发送给关心的微信好友
  9. springboot打包不同环境配置与shell脚本部署
  10. Abp + MongoDb 改造默认的审计日志存储位置
  11. 安全测试之Top 10 漏洞的分析
  12. C++系统自己主动生成默认构造函数的情况
  13. Linux内核分析第六次作业
  14. JUnit手动设计测试方法以及与Randoop的自动生成测试的比较
  15. [转]Java加密算法
  16. 57. 激活office时出下以下问题的解决方案
  17. php后台+前端开发过程整理
  18. LVS-net
  19. Django-2的路由层(URLconf)
  20. TXSQL:云计算时代数据库核弹头——云+未来峰会开发者专场回顾

热门文章

  1. python函数 全局变量和局部变量
  2. C++多线程基础学习笔记(三)
  3. 2-SAT问题介绍求解 + 模板题P4782
  4. 正则爬取某段子网站前20页段子(request库)
  5. 最大两队竞争值(暴力dfs)--牛客多校第二场
  6. 入职一个月快速熟悉大型Vue项目经验感想
  7. C数据结构排序算法——希尔排序法用法总结(转http://www.cnblogs.com/skywang12345/p/3597597.html)
  8. C# 连接 Oracle数据库增删改查,事务
  9. 机器学习-SVM-核函数
  10. luogu P3226 [HNOI2012]集合选数