利用es6 new.target来对模拟抽象类的方法


Posted in Javascript onMay 10, 2019

起源

最近在使用 Symbol 来做为唯一值,发现 Symbol 无法进行 new 操作,只能当作函数使用,只要进行了new 就会发生类型错误

new Symbol()

// error
Uncaught TypeError: Symbol is not a constructor
  at new Symbol (<anonymous>)
  at <anonymous>:1:1

在不考虑底层实现的情况下,在代码层面是否能够实现一个函数只可以进行调用而不可以进行 new 操作呢?思考之后如下写出:

function disConstructor() {
 if (this !== window) {
  throw new TypeError(' disConstructor is not a constructor')
 }
 console.log('gogo go')
}

// 测试结果如下
disConstructor() // gogo go

new disConstructor()

// error
Uncaught TypeError: disConstructor is not a constructor
  at new disConstructor (<anonymous>:3:15)
  at <anonymous>:1:1

如果使用 nodejs,window 可以切换为 global, 代码运行结果不变,因为对于个人而言没有适用场景。于是就没有继续研究下去,可是最近在从新翻阅 es6 时候发现 new.target这个属性。

new.target 属性

介绍(引用 mdn 文档)

new.target属性允许你检测函数或构造方法是否是通过new运算符被调用的。

在通过new运算符被初始化的函数或构造方法中,new.target返回一个指向构造方法或函数的引用。在普通的函数调用中,new.target 的值是undefined。

这样的话 我们的代码就可以这样改为

function disConstructor() {
 // 普通的函数调用中,new.target 的值是undefined。
 if (new.target) {
  throw new TypeError(' disConstructor is not a constructor')
 }
 console.log('gogo go')
}

得到与上述代码一样的答案。

深入

难道 es6 特地添加的功能仅仅只能用于检查一下我们的函数调用方式吗?

在查阅的过程各种发现了大多数都方案都是用 new.target 写出只能被继承的类。类似于实现java的抽象类。

class Animal {
 constructor(name, age) {
  if (new.target === Animal) {
   throw new Error('Animal class can`t instantiate');
  }
  this.name = name
  this.age = age
 }
 // 其他代码
 ...
}

class Dog extends Animal{
 constructor(name, age, sex) {
  super(name, age)
  this.sex = sex
 }
}

new Animal()
// error
Uncaught Error: Animal class can`t instantiate
  at new Animal (<anonymous>:4:13)
  at <anonymous>:1:1

new Dog('mimi', 12, '公')
// Dog {name: "mimi", age: 12, sex: "公"}

但是 java抽象类抽象方法需要重写,这个是没有方案的。于是在测试与使用的过程中,却意外发现了超类可以在构造期间访问派生类的原型,利用起来。

class Animal {
 constructor(name, age) {
  console.log(new.target.prototype)
 }
 // 其他代码
 ...
}

之前运行时调用需要重写的方法报错是这样写的。

class Animal {
 constructor(name, age) {
  this.name = name
  this.age = age
 }

 getName () {
  throw new Error('please overwrite getName method')
 }
}

class Dog extends Animal{
 constructor(name, age, sex) {
  super(name, age)
  this.sex = sex
 }
}

const pite = new Dog('pite', 2, '公')
a.getName()
// error
Uncaught Error: please overwrite getName method
  at Dog.getName (<anonymous>:8:11)
  at <anonymous>:1:3

然而此时利用 new.target ,我就可以利用 构造期间 对子类进行操作报错。

class Animal {
 constructor(name, age) {
  // 如果 target 不是 基类 且 没有 getName 报错
  if (new.target !== Animal && !new.target.prototype.hasOwnProperty('getName')) {
   throw new Error('please overwrite getName method')
  }
  this.name = name
  this.age = age
 }
}

class Dog extends Animal{
 constructor(name, age, sex) {
  super(name, age)
  this.sex = sex
 }
}

const pite = new Dog('pite', 2, '公')
// error
Uncaught Error: please overwrite getName method
  at new Animal (<anonymous>:5:13)
  at new Dog (<anonymous>:14:5)
  at <anonymous>:1:1

此时可以把运行方法时候发生的错误提前到构造时期,虽然都是在运行期,但是该错误触发机制要早危害要大。反而对代码是一种保护。

当然了,利用超类可以在构造期间访问派生类的原型作用远远不是那么简单,必然是很强大的,可以结合业务场景谈一谈理解和作用。

其他方案

  • 增加 编辑器插件
  • proxy
  • 修饰器

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
jquery实现的图片点击滚动效果
Apr 29 Javascript
用原生JS获取CLASS对象(很简单实用)
Oct 15 Javascript
js实现div弹出层的方法
Nov 20 Javascript
Javascript数据结构与算法之列表详解
Mar 12 Javascript
JS根据生日算年龄的方法
May 05 Javascript
js实现精美的银灰色竖排折叠菜单
May 16 Javascript
JavaScript对象数组排序函数及六个用法
Dec 23 Javascript
JavaScript实现给定时间相加天数的方法
Jan 25 Javascript
详解使用element-ui table组件的筛选功能的一个小坑
Nov 02 Javascript
爬虫利器Puppeteer实战
Jan 09 Javascript
Layui实现数据表格中鼠标悬浮图片放大效果,离开时恢复原图的方法
Sep 11 Javascript
JavaScript 作用域scope简单汇总
Oct 23 Javascript
Angular4.0动画操作实例详解
May 10 #Javascript
Angular 2使用路由自定义弹出组件toast操作示例
May 10 #Javascript
Angular2使用SVG自定义图表(条形图、折线图)组件示例
May 10 #Javascript
vue 实现搜索的结果页面支持全选与取消全选功能
May 10 #Javascript
Vue项目中配置pug解析支持
May 10 #Javascript
Angular2实现的秒表及改良版示例
May 10 #Javascript
node中IO以及定时器优先级详解
May 10 #Javascript
You might like
再次研究下cache_lite
2007/02/14 PHP
PHP 高级课程笔记 面向对象
2009/06/21 PHP
php 静态变量与自定义常量的使用方法
2010/01/26 PHP
php curl 伪造IP来源的实例代码
2012/11/01 PHP
YII中assets的使用示例
2014/07/31 PHP
详解PHP的Yii框架中扩展的安装与使用
2016/04/01 PHP
使用PHP连接多种数据库的实现代码(mysql,access,sqlserver,Oracle)
2016/12/21 PHP
当自定义数据属性为json格式字符串时jQuery的data api问题探讨
2013/02/18 Javascript
基于jQuery的判断iPad、iPhone、Android是横屏还是竖屏的代码
2014/05/11 Javascript
探寻Javascript执行效率问题
2014/11/12 Javascript
js表单验证实例讲解
2016/03/31 Javascript
JS基于递归实现倒计时效果的方法
2016/11/26 Javascript
微信小程序学习之数据处理详解
2017/07/05 Javascript
详解AngularJS跨页面传值(ui-router)
2017/08/23 Javascript
详解使用nvm管理多版本node的方法
2017/08/30 Javascript
详解PHP后期静态绑定分析与应用
2018/03/21 Javascript
vue项目打包后打开页面空白解决办法
2018/06/29 Javascript
JavaScript高级函数应用之分时函数实例分析
2018/08/03 Javascript
微信小程序实现的绘制table表格功能示例
2019/04/26 Javascript
jQuery实现高级检索功能
2019/05/28 jQuery
详解vue微信网页授权最终解决方案
2019/06/16 Javascript
uniapp 仿微信的右边下拉选择弹出框的实现代码
2020/07/12 Javascript
从源码解析Python的Flask框架中request对象的用法
2016/06/02 Python
Python生成数字图片代码分享
2017/10/31 Python
Python3简单实现串口通信的方法
2019/06/12 Python
浅谈Python访问MySQL的正确姿势
2020/01/07 Python
详解有关PyCharm安装库失败的问题的解决方法
2020/02/02 Python
css3 矩阵的使用详解
2018/03/20 HTML / CSS
加拿大在线隐形眼镜专家:PerfectLens.ca
2016/11/19 全球购物
澳大利亚领先的宠物用品商店:VetSupply
2017/09/08 全球购物
上学迟到的检讨书
2014/01/11 职场文书
企业口号大全
2014/06/12 职场文书
鼓舞士气的口号
2014/06/16 职场文书
迎国庆演讲稿
2014/09/05 职场文书
2016关于军训的心得体会
2016/01/11 职场文书
php 解析非标准json、非规范json
2021/04/01 PHP