利用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 相关文章推荐
IE6-8中Date不支持toISOString的修复方法
May 04 Javascript
详解angularjs利用ui-route异步加载组件
May 21 Javascript
JS实现点击下拉菜单把选择的内容同步到input输入框内的实例
Jan 23 Javascript
Vue项目引发的「过滤器」使用教程
Mar 12 Javascript
微信小程序学习笔记之跳转页面、传递参数获得数据操作图文详解
Mar 28 Javascript
从0到1搭建Element的后台框架的方法步骤
Apr 10 Javascript
vue-cli脚手架引入弹出层layer插件的几种方法
Jun 24 Javascript
浅谈vue项目用到的mock数据接口的两种方式
Oct 09 Javascript
Vue混入mixins滚动触底的方法
Nov 22 Javascript
使用vue打包进行云服务器上传的问题
Mar 02 Javascript
通过实例解析JavaScript for in及for of区别
Jun 15 Javascript
JS事件循环机制event loop宏任务微任务原理解析
Aug 04 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
杏林同学录(一)
2006/10/09 PHP
PHP 输出缓存详解
2009/06/20 PHP
php学习笔记之 函数声明(二)
2011/06/09 PHP
jquery对表单操作2
2011/04/06 Javascript
formValidator3.3的ajaxValidator一些异常分析
2011/07/12 Javascript
一个基于jQuery的树型插件(OrangeTree)使用介绍
2012/05/03 Javascript
javascript返回顶部效果(自写代码)
2013/01/06 Javascript
使用jQuery和PHP实现类似360功能开关效果
2014/02/12 Javascript
用js通过url传参把数据从一个页面传到另一个页面
2014/09/01 Javascript
JavaScript实现按Ctrl键打开新页面
2014/09/04 Javascript
Javascript基础教程之for循环
2015/01/18 Javascript
JavaScript中定义类的方式详解
2016/01/07 Javascript
JAVA中截取字符串substring用法详解
2017/04/14 Javascript
jQuery取得元素标签名称小结(附代码)
2017/08/16 jQuery
在JavaScript中实现链式调用的实现
2019/12/24 Javascript
谈谈JavaScript令人迷惑的==与+
2020/08/31 Javascript
Python排序搜索基本算法之选择排序实例分析
2017/12/09 Python
python模块之paramiko实例代码
2018/01/31 Python
Django通过dwebsocket实现websocket的例子
2019/11/15 Python
pytorch 自定义卷积核进行卷积操作方式
2019/12/30 Python
python用TensorFlow做图像识别的实现
2020/04/21 Python
Keras自定义IOU方式
2020/06/10 Python
利用django创建一个简易的博客网站的示例
2020/09/29 Python
python如何利用Mitmproxy抓包
2020/10/10 Python
python爬虫请求头的使用
2020/12/01 Python
复古服装:RetroStage
2019/05/10 全球购物
Tomcat中怎么使用log4j输出所有的log
2016/07/07 面试题
某/etc/fstab文件中的某行如下: /dev/had5 /mnt/dosdata msdos defaults,usrquota 1 2 请解释其含义
2013/04/11 面试题
工商管理本科毕业生求职信范文
2013/10/05 职场文书
饮料业务员岗位职责
2013/12/15 职场文书
赔偿协议书范本
2014/09/12 职场文书
开展党的群众路线教育实践活动领导班子对照检查材料
2014/09/25 职场文书
2015年度高中教师工作总结
2015/05/26 职场文书
PHP中->和=>的意思
2021/03/31 PHP
pandas提升计算效率的一些方法汇总
2021/05/30 Python
Java 超详细讲解ThreadLocal类的使用
2022/04/07 Java/Android