ES6类和继承的实现原理(代码示例)

本篇文章给大家带来的内容是关于es6继承的实现原理(代码示例),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。

1、es6 class 使用

javascript使用的是原型式继承,我们可以通过原型的特性实现类的继承,
es6为我们提供了像面向对象继承一样的语法糖。

class Parent {  constructor(a){    this.filed1 = a;  }  filed2 = 2;  func1 = function(){}}class Child extends Parent {    constructor(a,b) {      super(a);      this.filed3 = b;    }    filed4 = 1;  func2 = function(){}}

登录后复制

下面我们借助babel来探究es6类和继承的实现原理。

1.类的实现

转换前:

class Parent {  constructor(a){    this.filed1 = a;  }  filed2 = 2;  func1 = function(){}}

登录后复制

转换后:

function _classCallCheck(instance, Constructor) {  if (!(instance instanceof Constructor)) {    throw new TypeError("Cannot call a class as a function");  }}var Parent = function Parent(a) {  _classCallCheck(this, Parent);  this.filed2 = 2;  this.func1 = function () { };  this.filed1 = a;};

登录后复制

可见class的底层依然是构造函数:

1.调用_classCallCheck方法判断当前函数调用前是否有new关键字。

构造函数执行前有new关键字,会在构造函数内部创建一个空对象,将构造函数的proptype指向这个空对象的_proto_,并将this指向这个空对象。如上,_classCallCheck中:this instanceof Parent 返回true。

若构造函数前面没有new则构造函数的proptype不会不出现在this的原型链上,返回false。

2.将class内部的变量和函数赋给this

3.执行constuctor内部的逻辑

4.return this (构造函数默认在最后我们做了)。

2.继承实现

转换前:

class Child extends Parent {    constructor(a,b) {      super(a);      this.filed3 = b;    }    filed4 = 1;  func2 = function(){}}

登录后复制

转换后:

我们先看Child内部的实现,再看内部调用的函数是怎么实现的:

var Child = function (_Parent) {  _inherits(Child, _Parent);  function Child(a, b) {    _classCallCheck(this, Child);    var _this = _possibleConstructorReturn(this, (Child.__proto__ || Object.getPrototypeOf(Child)).call(this, a));    _this.filed4 = 1;    _this.func2 = function () {};    _this.filed3 = b;    return _this;  }  return Child;}(Parent);

登录后复制

1.调用_inherits函数继承父类的proptype。

_inherits内部实现:

function _inherits(subClass, superClass) {  if (typeof superClass !== "function" && superClass !== null) {    throw new TypeError("Super expression must either be null or a function, not " + typeof superClass);  }  subClass.prototype = Object.create(superClass && superClass.prototype, {    constructor: { value: subClass, enumerable: false, writable: true, configurable: true }  });  if (superClass)    Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass;}

登录后复制

(1) 校验父构造函数。

(2) 典型的寄生继承:用父类构造函数的proptype创建一个空对象,并将这个对象指向子类构造函数的proptype。

(3) 将父构造函数指向子构造函数的_proto_(这步是做什么的不太明确,感觉没什么意义。)

2.用一个闭包保存父类引用,在闭包内部做子类构造逻辑。

3.new检查。

4.用当前this调用父类构造函数。

var _this = _possibleConstructorReturn(this, (Child.__proto__ || Object.getPrototypeOf(Child)).call(this, a));

登录后复制

这里的Child.__proto__ || Object.getPrototypeOf(Child)实际上是父构造函数(_inherits最后的操作),然后通过call将其调用方改为当前this,并传递参数。(这里感觉可以直接用参数传过来的Parent)

function _possibleConstructorReturn(self, call) {  if (!self) {    throw new ReferenceError("this hasn't been initialised - super() hasn't been called");  }  return call && (typeof call === "object" || typeof call === "function") ? call : self;}

登录后复制

校验this是否被初始化,super是否调用,并返回父类已经赋值完的this。

5.将行子类class内部的变量和函数赋给this。

6.执行子类constuctor内部的逻辑。

可见,es6实际上是为我们提供了一个“组合寄生继承”的简单写法。

3. super

super代表父类构造函数。

super.fun1() 等同于 Parent.fun1()  或 Parent.prototype.fun1()。

super() 等同于Parent.prototype.construtor()

当我们没有写子类构造函数时:

var Child = function (_Parent) {  _inherits(Child, _Parent);  function Child() {    _classCallCheck(this, Child);    return _possibleConstructorReturn(this, (Child.__proto__ || Object.getPrototypeOf(Child)).apply(this, arguments));  }  return Child;}(Parent);

登录后复制

可见默认的构造函数中会主动调用父类构造函数,并默认把当前constructor传递的参数传给了父类。

所以当我们声明了constructor后必须主动调用super(),否则无法调用父构造函数,无法完成继承。

典型的例子就是Reatc的Component中,我们声明constructor后必须调用super(props),因为父类要在构造函数中对props做一些初始化操作。

以上就是ES6类和继承的实现原理(代码示例)的详细内容,更多请关注【创想鸟】其它相关文章!

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

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

(0)
上一篇 2025年3月8日 01:07:00
下一篇 2025年3月8日 01:07:04

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

相关推荐

  • JavaScript如何判断json中数据的真假(代码示例)

    本篇文章给大家带来的内容是关于javascript如何判断json中数据的真假(代码示例),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 在项目开发过程中,很多时候需要处理json数据,在处理的过程中,难免会对数据进行真假…

    编程技术 2025年3月8日
    100
  • JavaScript原型和原型链的方法介绍(代码示例)

    本篇文章给大家带来的内容是关于javascript原型和原型链的方法介绍(代码示例),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 1、题目 如何准确判断一个变量是数组 写一个原型链继承的例子 继承实现的其他方式 立即学习…

    编程技术 2025年3月8日
    200
  • Vue的HOC技术如何开发一个无限加载列表(代码示例)

    本篇文章给大家带来的内容是关于vue的hoc技术如何开发一个无限加载列表(代码示例),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 在web开发上,我们都对数据采用分页加载的机制,一种变形就是在页面采用循环加载的机制,拉到…

    2025年3月8日
    200
  • JavaScript中回调是是什么

    回调是在完成其他事情之后处理某事的好方法;如果我们想在执行函数后立即执行另一个函数,就可以使用回调。下面本篇文章就来带大家认识一下JavaScript的回调,希望对大家有所帮助。 JavaScript函数具有Objects类型。所以,就像任…

    2025年3月8日
    200
  • HTTPS如何保证Web安全?(代码示例)

    本篇文章给大家带来的内容是关于https如何保证web安全?(代码示例),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 HTTPS(全称:HyperText Transfer Protocol over Secure So…

    2025年3月8日 编程技术
    200
  • JavaScript中去抖与节流的详细介绍(代码示例)

    本篇文章给大家带来的内容是关于javascript中去抖与节流的详细介绍(代码示例),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 无论是面试还是在讨论浏览器优化过程中,都会涉及到去抖动和节流的问题。总的来说,这二者是一种…

    编程技术 2025年3月8日
    200
  • AJAX跨域的详细介绍(代码示例)

    本篇文章给大家带来的内容是关于ajax跨域的详细介绍(代码示例),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 AJAX 的跨域问题,常用的解决方法有两种,简单记录下,详细内容查看参考资料。以下示例自建两个域名测试,www…

    编程技术 2025年3月8日
    200
  • 原生js实现移动端Touch滑动反弹的方法(代码示例)

    本篇文章给大家带来的内容是关于原生js实现移动端touch滑动反弹的方法(代码示例),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 移动端 Touch 滑动反弹 什么是Touch滑动?就是类似于PC端的滚动事件,但是在移动…

    2025年3月8日 编程技术
    200
  • ES6对象的扩展及新增方法的内容总结(附示例)

    本篇文章给大家带来的内容是关于es6对象的扩展及新增方法的内容总结(附示例),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 1、属性的简洁表示法 ES6允许直接写入变量和函数,作为对象的属性和方法。这样的书写更加简洁。 c…

    编程技术 2025年3月8日
    200
  • Javascript隐式转换怎么用?(代码示例)

    本篇文章给大家带来的内容是关于javascript隐式转换怎么用?(代码示例),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 确定两个变量是否相等是编程中的一个非常重要的操作。在比较字符串、数值和布尔值的相等性时,问题还比…

    编程技术 2025年3月8日
    200

发表回复

登录后才能评论