利用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 相关文章推荐
js文件中调用js的实现方法小结
Oct 23 Javascript
js 刷新页面的代码小结 推荐
Apr 02 Javascript
jQuery hover 延时器实现代码
Mar 12 Javascript
jQuery过滤选择器:not()方法使用介绍
Apr 20 Javascript
jquery、js调用iframe父窗口与子窗口元素的方法整理
Jul 31 Javascript
React.js入门学习第一篇
Mar 30 Javascript
js 动态生成html 触发事件传参字符转义的实例
Feb 14 Javascript
Angular 4依赖注入学习教程之组件服务注入(二)
Jun 04 Javascript
JS实现按钮添加背景音乐示例代码
Oct 17 Javascript
JS实现数组去重及数组内对象去重功能示例
Feb 02 Javascript
简单了解JavaScript中的执行上下文和堆栈
Jun 24 Javascript
jQuery实现鼠标移入显示蒙版效果
Jan 11 jQuery
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
php版阿里云OSS图片上传类详解
2016/12/01 PHP
php设计模式之代理模式分析【星际争霸游戏案例】
2020/03/23 PHP
jQuery-onload让第一次页面加载时图片是淡入方式显示
2012/05/23 Javascript
javascript常见用法总结
2014/05/22 Javascript
JavaScript导出Excel实例详解
2014/11/25 Javascript
js显示文本框提示文字的方法
2015/05/07 Javascript
浅谈jQuery构造函数分析
2015/05/11 Javascript
js实现鼠标划过给div加透明度的方法
2015/05/25 Javascript
JS获取复选框的值,并传递到后台的实现方法
2016/05/30 Javascript
JavaScript作用域示例详解
2016/07/07 Javascript
jQuery实现的图片轮播效果完整示例
2016/09/12 Javascript
js实现简易聊天对话框
2017/08/17 Javascript
有趣的JavaScript隐式类型转换操作实例分析
2020/05/02 Javascript
原生js拖拽功能制作滑动条实例代码
2021/02/05 Javascript
微信小程序input抖动问题的修复方法
2021/03/03 Javascript
python实现百万答题自动百度搜索答案
2018/01/16 Python
matplotlib savefig 保存图片大小的实例
2018/05/24 Python
python将txt文件读入为np.array的方法
2018/10/30 Python
python高斯分布概率密度函数的使用详解
2019/07/10 Python
Python Django框架模板渲染功能示例
2019/11/08 Python
基于Python获取城市近7天天气预报
2019/11/26 Python
python switch 实现多分支选择功能
2020/12/21 Python
Clarks鞋美国官网:全球领军鞋履品牌
2017/05/13 全球购物
Lacoste(法国鳄鱼)加拿大官网:以标志性的POLO衫而闻名
2019/05/15 全球购物
区域销售经理岗位职责
2013/12/10 职场文书
人事档案接收函
2014/01/12 职场文书
高中生的自我鉴定范文
2014/01/24 职场文书
微观物理专业自荐信
2014/01/26 职场文书
生物制药专业自我鉴定
2014/02/19 职场文书
合伙经营协议书
2014/04/18 职场文书
检察机关个人对照检查材料
2014/09/15 职场文书
银行奉献演讲稿
2014/09/16 职场文书
乡镇党建工作总结2015
2015/05/19 职场文书
JavaScript控制台的更多功能
2021/04/28 Javascript
Python中基础数据类型 set集合知识点总结
2021/08/02 Python
详解JS数组方法
2021/11/20 Javascript