对于JS继承详细介绍( 原型链,构造函数,组合,原型式,寄生式,寄生组合,Class extends)

这篇文章给大家分享的内容是关于对js继承详细介绍( 原型链,构造函数,组合,原型式,寄生式,寄生组合,class extends),有一定的参考价值,有需要的朋友可以参考一下。

说实在话,以前我只需要知道“寄生组合继承”是最好的,有个祖传代码模版用就行。最近因为一些事情,几个星期以来一直心心念念想整理出来。本文以《JavaScript高级程序设计》上的内容为骨架,补充了ES6 Class的相关内容,从我认为更容易理解的角度将继承这件事叙述出来,希望大家能有所收获。

1. 继承分类

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

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

下面马上进入正题。

clipboard.png

2. 继承方式

上图上半区的原型链继承,构造函数继承,组合继承,网上内容比较多,本文不作详细描述,只指出重点。这里给出了我认为最容易理解的一篇《JS中的继承(上)》。如果对上半区的内容不熟悉,可以先看这篇文章,再回来继续阅读;如果已经比较熟悉,这部分可以快速略过。另,上半区大量借用了yq前端的一篇继承文章[1]。

2.1 原型式继承

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

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

登录后复制

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

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

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

2.2 构造函数继承

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

SuperType.call(SubType);

登录后复制

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

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

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

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

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

登录后复制

优点:

父类的方法可以被复用

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

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

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

2.4 原型式继承

核心:原型式继承的object方法本质上是对参数对象的一个浅复制。
优点:父类方法可以复用
缺点:

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

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

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继承的语法糖
不同点:

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

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

3. 总结

ES6 Class extends是ES5继承的语法糖

JS的继承除了构造函数继承之外都基于原型链构建的

可以用寄生组合继承实现ES6 Class extends,但是还是会有细微的差别

相关推荐:

Vue中钩子函数的具体介绍

AngularJs自定义指令可以如何来设置以及自定义指令的命名规范

以上就是对于JS继承详细介绍( 原型链,构造函数,组合,原型式,寄生式,寄生组合,Class extends)的详细内容,更多请关注【创想鸟】其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至253000106@qq.com举报,一经查实,本站将立刻删除。

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2745511.html

(0)
上一篇 2025年3月8日 03:45:40
下一篇 2025年2月25日 08:51:33

AD推荐 黄金广告位招租... 更多推荐

相关推荐

  • Vue中钩子函数的具体介绍

    这篇文章给大家分享的内容是关于vue的钩子函数[路由导航守卫、keep-alive、生命周期钩子] ,有一定的参考价值,有需要的朋友可以参考一下。 前言 说到Vue的钩子函数,可能很多人只停留在一些很简单常用的钩子(created,moun…

    编程技术 2025年3月8日
    000
  • Vue常用的三种传值方式

    这篇文章给大家分享的内容是关于vue常用的三种传值方式,有一定的参考价值,有需要的朋友可以参考一下。 父组件向子组件进行传值: 父组件: 父组件: import child from ‘./child’ export default { c…

    编程技术 2025年3月8日
    200
  • js中找到两个链表的第一个公共结点的算法

    这篇文章给大家分享的内容是关于js中找到两个链表的第一个公共结点的算法  ,有一定的参考价值,有需要的朋友可以参考一下。 题目描述 输入两个链表,找出它们的第一个公共结点。 分析 考虑到两个链表不一样长的情况,算出两个链表的长度差,然后长的…

    编程技术 2025年3月8日
    200
  • 对React事件系统的解析

    这篇文章给大家分享的内容是关于对react事件系统的解析,有一定的参考价值,有需要的朋友可以参考一下。 一 前言 React事件系统有两类:合成事件和原生事件。在写React组件是我们很容易绑定一个合成事件,但是在一个组件里面是没有办法去绑…

    2025年3月8日
    200
  • js的模块化分析(命名空间)

    这篇文章给大家分享的内容是关于js的模块化分析(命名空间),有一定的参考价值,有需要的朋友可以参考一下。 代码组织到模块中,为了让代码根据模块化,即,不同场景中的模块复用。模块是一个独立的js文件,模块文件可以包含一个类定义,一组相关的类,…

    2025年3月8日
    200
  • js中如何将call及apply和bind用原生来实现

    这篇文章给大家分享的内容是关于js中如何将call及apply和bind用原生来实现,有一定的参考价值,有需要的朋友可以参考一下。 因为关乎到了this指向的问题,call、apply和bind的用法可以说是老生常谈了。这篇文章的主要作用是…

    编程技术 2025年3月8日
    200
  • 关于TypeScript在node项目中的实践分析

    这篇文章给大家分享的内容是关于typescript在node项目中的实践分析,有一定的参考价值,有需要的朋友可以参考一下。 TypeScript可以理解为是JavaScript的一个超集,也就是说涵盖了所有JavaScript的功能,并在之…

    2025年3月8日 编程技术
    200
  • 对Angular 6中滚动列表组件的封装的分析

    这篇文章给大家分享的内容是关于对angular 6中滚动列表组件的封装的分析,有一定的参考价值,有需要的朋友可以参考一下。 前言 学习应为input和output相结合之过程,这就是写这篇文章的原因。在大屏幕展示web APP中,经常会用到…

    编程技术 2025年3月8日
    200
  • react-navigation在开发中的使用介绍

    这篇文章给大家分享的内容是关于react-navigation在开发中的使用介绍,有一定的参考价值,有需要的朋友可以参考一下。 日常废话 react-navigation是一个来源于react社区的导航解决方案。 以我一个月资深的react…

    编程技术 2025年3月8日
    200
  • 如何使用node生成验证码

    这篇文章给大家分享的内容是关于如何使用node生成验证码,有一定的参考价值,有需要的朋友可以参考一下。 前言 网络安全始终是一个重要话题,比如当你发现有人在恶意请求你的网站的邮箱注册接口,那么你可以考虑在服务端加上验证码,提高网站的安全性,…

    2025年3月8日
    200

发表回复

登录后才能评论