位置: 编程技术 - 正文

ES6 javascript中Class类继承用法实例详解

编辑:rootadmin

推荐整理分享ES6 javascript中Class类继承用法实例详解,希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:,内容如对您有帮助,希望把文章链接给更多的朋友!

本文实例讲述了ES6 javascript中Class类继承用法。分享给大家供大家参考,具体如下:

1. 基本用法

Class 之间可以通过extends关键字实现继承, 这比 ES5 的通过修改原型链实现继承, 要清晰和方便很多。

上面代码定义了一个ColorPoint类, 该类通过extends关键字, 继承了Point类的所有属性和方法。 但是由于没有部署任何代码, 所以这两个类完全一样, 等于复制了一个Point类。 下面, 我们在ColorPoint内部加上代码。

上面代码中, constructor方法和toString方法之中, 都出现了super关键字, 它在这里表示父类的构造函数, 用来新建父类的this对象。

子类必须在constructor方法中调用super方法, 否则新建实例时会报错。 这是因为子类没有自己的this对象, 而是继承父类的this对象, 然后对其进行加工。 如果不调用super方法, 子类就得不到this对象。

上面代码中, ColorPoint继承了父类Point, 但是它的构造函数没有调用super方法, 导致新建实例时报错。

ES5 的继承, 实质是先创造子类的实例对象this, 然后再将父类的方法添加到this上面( Parent.apply(this))。 ES6 的继承机制完全不同, 实质是先创造父类的实例对象this( 所以必须先调用super方法), 然后再用子类的构造函数修改this。

如果子类没有定义constructor方法, 这个方法会被默认添加, 代码如下。 也就是说, 不管有没有显式定义, 任何一个子类都有constructor方法。

另一个需要注意的地方是, 在子类的构造函数中, 只有调用super之后, 才可以使用this关键字, 否则会报错。 这是因为子类实例的构建, 是基于对父类实例加工, 只有super方法才能返回父类实例。

上面代码中, 子类的constructor方法没有调用super之前, 就使用this关键字, 结果报错, 而放在super方法之后就是正确的。

下面是生成子类实例的代码。

上面代码中, 实例对象cp同时是ColorPoint和Point两个类的实例, 这与 ES5 的行为完全一致。

2. 类的 prototype 属性和 __proto__ 属性

大多数浏览器的 ES5 实现之中, 每一个对象都有__proto__属性, 指向对应的构造函数的 prototype 属性。 Class 作为构造函数的语法糖, 同时有prototype 属性和__proto__属性, 因此同时存在两条继承链。

( 1) 子类的__proto__属性, 表示构造函数的继承, 总是指向父类。( 2) 子类prototype属性的__proto__属性, 表示方法的继承, 总是指向父类的prototype属性。

上面代码中, 子类B的__proto__属性指向父类A, 子类B的prototype属性的__proto__属性指向父类A的prototype属性。

这样的结果是因为, 类的继承是按照下面的模式实现的。

《对象的扩展》 一章给出过Object.setPrototypeOf方法的实现。

因此, 就得到了上面的结果。

这两条继承链, 可以这样理解: 作为一个对象, 子类( B) 的原型( __proto__属性) 是父类( A); 作为一个构造函数, 子类( B) 的原型( prototype属性) 是父类的实例。

3. Extends 的继承目标

extends关键字后面可以跟多种类型的值。

上面代码的A, 只要是一个有prototype属性的函数, 就能被B继承。 由于函数都有prototype属性( 除了Function.prototype函数), 因此A可以是任意函数。

下面, 讨论三种特殊情况。

第一种特殊情况, 子类继承 Object 类。

这种情况下, A其实就是构造函数Object的复制, A的实例就是Object的实例。

第二种特殊情况, 不存在任何继承。

这种情况下, A 作为一个基类( 即不存在任何继承), 就是一个普通函数, 所以直接继承Funciton.prototype。 但是, A调用后返回一个空对象( 即Object实例), 所以A.prototype.__proto__指向构造函数( Object) 的prototype属性。

第三种特殊情况, 子类继承null。

ES6 javascript中Class类继承用法实例详解

这种情况与第二种情况非常像。 A也是一个普通函数, 所以直接继承Funciton.prototype。 但是, A 调用后返回的对象不继承任何方法, 所以它的__proto__指向Function.prototype, 即实质上执行了下面的代码。

4. Object.getPrototypeOf()

Object.getPrototypeOf方法可以用来从子类上获取父类。

因此, 可以使用这个方法判断, 一个类是否继承了另一个类。

5. super 关键字

super这个关键字, 有两种用法, 含义不同。

( 1) 作为函数调用时( 即super(...args)), super代表父类的构造函数。

( 2) 作为对象调用时( 即super.prop或super.method()), super代表父类。 注意, 此时super即可以引用父类实例的属性和方法, 也可以引用父类的静态方法。

上面代码中, 子类通过super关键字, 调用父类实例的_p属性。由于, 对象总是继承其他对象的, 所以可以在任意一个对象中, 使用super关键字。

6. 实例的 __proto__ 属性

子类实例的 __proto__ 属性的 __proto__ 属性, 指向父类实例的 __proto__ 属性。 也就是说, 子类的原型的原型, 是父类的原型。

上面代码中, ColorPoint继承了Point, 导致前者原型的原型是后者的原型。

因此, 通过子类实例的__proto__.__proto__属性, 可以修改父类实例的行为。

上面代码在ColorPoint的实例p2上向Point类添加方法, 结果影响到了Point的实例p1。

原生构造函数的继承

原生构造函数是指语言内置的构造函数, 通常用来生成数据结构。 ECMAScript 的原生构造函数大致有下面这些。

以前, 这些原生构造函数是无法继承的, 比如, 不能自己定义一个Array的子类。

上面代码定义了一个继承 Array 的MyArray类。 但是, 这个类的行为与Array完全不一致。

之所以会发生这种情况, 是因为子类无法获得原生构造函数的内部属性, 通过Array.apply() 或者分配给原型对象都不行。 原生构造函数会忽略apply方法传入的this, 也就是说, 原生构造函数的this无法绑定, 导致拿不到内部属性。

ES5 是先新建子类的实例对象this, 再将父类的属性添加到子类上, 由于父类的内部属性无法获取, 导致无法继承原生的构造函数。 比如, Array 构造函数有一个内部属性[[DefineOwnProperty]], 用来定义新属性时, 更新length属性, 这个内部属性无法在子类获取, 导致子类的length属性行为不正常。

下面的例子中, 我们想让一个普通对象继承Error对象。

上面代码中, 我们想通过Error.call(e) 这种写法, 让普通对象e具有Error对象的实例属性。 但是, Error.call() 完全忽略传入的第一个参数, 而是返回一个新对象, e本身没有任何变化。 这证明了Error.call(e) 这种写法, 无法继承原生构造函数。

ES6 允许继承原生构造函数定义子类, 因为 ES6 是先新建父类的实例对象this, 然后再用子类的构造函数修饰this, 使得父类的所有行为都可以继承。 下面是一个继承Array的例子。

上面代码定义了一个MyArray类, 继承了Array构造函数, 因此就可以从MyArray生成数组的实例。 这意味着, ES6 可以自定义原生数据结构( 比如Array、 String 等) 的子类, 这是 ES5 无法做到的。

上面这个例子也说明, extends关键字不仅可以用来继承类, 还可以用来继承原生的构造函数。 因此可以在原生数据结构的基础上, 定义自己的数据结构。 下面就是定义了一个带版本功能的数组。

上面代码中, VersionedArray结构会通过commit方法, 将自己的当前状态存入history属性, 然后通过revert方法, 可以撤销当前版本, 回到上一个版本。 除此之外, VersionedArray依然是一个数组, 所有原生的数组方法都可以在它上面调用。

下面是一个自定义Error子类的例子。

注意, 继承Object的子类, 有一个行为差异。

上面代码中, NewObj继承了Object, 但是无法通过super方法向父类Object传参。 这是因为 ES6 改变了Object构造函数的行为, 一旦发现Object方法不是通过new Object() 这种形式调用, ES6 规定Object构造函数会忽略参数。

更多相关内容可查看本站专题:《ECMAScript6(ES6)入门教程》、《JavaScript数组操作技巧总结》、《JavaScript字符与字符串操作技巧总结》、《JavaScript数据结构与算法技巧总结》、《JavaScript错误与调试技巧总结》及《javascript面向对象入门教程》

希望本文所述对大家基于ECMAScript的程序设计有所帮助。

JavaScript Date对象应用实例分享 本文实例为大家分享了jsDate对象应用3个实例,供大家参考,具体内容如下一.获取日期时间,秒数实时跳动!DOCTYPEhtmlhtmllang="en"headmetacharset="UTF-8"titledate/

JavaScript数组push方法使用注意事项 js数组的push方法,想必大家都知道是向数组末尾添加元素,但是有一个很关键的点需注意:引自MDN返回值当调用该方法时,新的length属性值将被返回。va

微信小程序商品详情页规格属性选择示例代码 detail.wxml展示页面!--轮播图--swiperclass="swiper"indicator-dots="{{indicatorDots}}"autoplay="{{autoplay}}"interval="{{interval}}"duration="{{duration}}"circular="{{circular}}"blockwx:for="{{pi

标签: ES6 javascript中Class类继承用法实例详解

本文链接地址:https://www.jiuchutong.com/biancheng/369678.html 转载请保留说明!

上一篇:移动端网页开发调试神器Eruda的介绍与使用技巧(移动端网页开发技术)

下一篇:JavaScript Date对象应用实例分享(javascript date format)

  • 购买的携税宝费用怎么做账?
  • 增值税的账务处理 月末怎么结转 案例
  • 转账支票开给个人收款人怎么写
  • 给客户提供的价值
  • 矿泉水资源税计税依据
  • 出租车行业怎么银行贷款
  • 出售商标使用权收入计入什么科目
  • 电子承兑必须对账吗
  • 接收虚开增值税专用发票要判刑吗
  • 建筑业附加税税率是多少
  • 小规模应税行为扣除额计算
  • 小规模纳税人税收政策2023
  • 建筑安装业跨省经营管理税务通知
  • 做季报和月报增发的区别
  • 汽车装饰用品大全进货
  • 现金预算在企业财务管理中是何地位
  • 汽车发票抵扣联需要盖章吗
  • 应付账款借方余额怎么平账
  • 个人其他应收款在贷方表示什么
  • 客户罚款记哪个科目
  • 苹果客服人工24小时
  • php注册功能的实现
  • 未实际发生的费用 可否申请赔偿
  • 经营性存款人违反规定
  • 前后端交互用什么技术
  • vue3 + ts
  • jupyter用法
  • vue怎么打断点
  • vue3当中如何监听新增的属性
  • 图像超分辨率重建数据集
  • html多页面
  • 发票融资会计处理
  • 行政性收费和事业性收费 财政局
  • 查补收入是否享受免税政策
  • Pythonround函数作用
  • 在mysql中设置事务保存点
  • 留抵进项税太多怎么办
  • 收据能否入账
  • 公司法人往公司账户打钱怎么能换回来
  • 固定资产报废时,后续未折旧额计入哪里
  • 进项留底是什么科目
  • 研发费用加计扣除是什么意思啊
  • 补开以前年度发票
  • 预付货款用什么会计科目
  • 公司购买黄金送客户可以抵税吗
  • 差旅费包括哪些费用
  • 员工异地工作
  • 酒店会计的账务处理
  • 注册资本可以随便填吗
  • 股权变更需要多久
  • 小企业建账流程图
  • mysql rand整数
  • mysql登录失败处理
  • sqlserver判断数据库是否存在
  • ubuntu拨号上网设置
  • ubuntu20开机自启动
  • linux系统怎么添加文件
  • linuxnamespace入门
  • win7断电后无法正常启动
  • win7系统纯净版和旗舰版
  • 电脑ahci模式什么意思
  • 判断div滑动到底怎么操作
  • opengl导入obj能动起来吗
  • nodejs基本原理
  • xbox无法连接无线网络
  • vue实现标签页效果
  • android遇到的难题,怎么解决的
  • Python中str is not callable问题详解及解决办法
  • ansible客户端需要装python
  • 完美世界3v3
  • input限制数字大小
  • First Class: UI of Android
  • 税务系统今天不能用
  • 企业安置几级残疾人
  • 广东省税务总局官网
  • 支付宝申领失业金申请审核多久
  • 北京国家税务局总局官网
  • 新疆医保哪里查
  • 三门峡哪个小区是原火葬场
  • 怎样登录市地税局网站
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设