利用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 相关文章推荐
javascript 控制input只允许输入的各种指定内容
Jun 19 Javascript
jquery实现的仿天猫侧导航tab切换效果
Aug 24 Javascript
基于javascript bootstrap实现生日日期联动选择
Apr 07 Javascript
JavaScript Canvas绘制圆形时钟效果
Aug 20 Javascript
jQuery实现的下雪动画效果示例【附源码下载】
Feb 02 jQuery
Vue 路由 过渡动效 数据获取方法
Jul 31 Javascript
React+Webpack快速上手指南(小结)
Aug 15 Javascript
Vue.js 十五分钟入门图文教程
Sep 12 Javascript
vue-cli3配置与跨域处理方法
Aug 17 Javascript
JavaScript变量Dom对象的所有属性
Apr 30 Javascript
解决vue+webpack项目接口跨域出现的问题
Aug 10 Javascript
Openlayers测量距离与面积的实现方法
Sep 25 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
php中处理mysql_fetch_assoc返回来的数组 不用foreach----echo
2011/05/04 PHP
PHP中使用sleep造成mysql读取失败的案例和解决方法
2014/08/21 PHP
php中删除数组的第一个元素和最后一个元素的函数
2015/03/07 PHP
php投票系统之增加与删除投票(管理员篇)
2016/07/01 PHP
js不能跳转到上一页面的问题解决方法
2013/03/01 Javascript
JavaScript和CSS通过expression实现Table居中显示
2013/06/28 Javascript
浅析javascript中函数声明和函数表达式的区别
2015/02/15 Javascript
详解jQuery Mobile自定义标签
2016/01/06 Javascript
实例讲解jQuery EasyUI tree中state属性慎用
2016/04/01 Javascript
Bootstrap文件上传组件之bootstrap fileinput
2016/11/25 Javascript
20行JS代码实现网页刮刮乐效果
2017/06/23 Javascript
AngularJS使用ocLazyLoad实现js延迟加载
2017/07/05 Javascript
js实现从左向右滑动式轮播图效果
2017/07/07 Javascript
Node.js中使用mongoose操作mongodb数据库的方法
2017/09/12 Javascript
JavaScript分步实现一个出生日期的正则表达式
2018/03/22 Javascript
用vscode开发vue应用的方法步骤
2019/05/06 Javascript
vue配置文件实现代理v2版本的方法
2019/06/21 Javascript
详解Nuxt内导航栏的两种实现方式
2020/04/16 Javascript
VUE中setTimeout和setInterval自动销毁案例
2020/09/07 Javascript
[01:18:36]LGD vs VP Supermajor 败者组决赛 BO3 第一场 6.10
2018/07/04 DOTA
Python装饰器decorator用法实例
2014/11/10 Python
简单谈谈Python中的反转字符串问题
2016/10/24 Python
python制作爬虫爬取京东商品评论教程
2016/12/16 Python
基于Django filter中用contains和icontains的区别(详解)
2017/12/12 Python
Python字典及字典基本操作方法详解
2018/01/30 Python
通过 Django Pagination 实现简单分页功能
2019/11/11 Python
英国灯具和灯泡网上商店:Lights.co.uk
2018/02/02 全球购物
String是最基本的数据类型吗?
2013/06/13 面试题
Unix如何在一行中运行多个命令
2015/05/29 面试题
实习自我鉴定范文
2013/10/30 职场文书
师德师风自我评价范文
2014/09/11 职场文书
2016党校学习心得体会范文
2016/01/07 职场文书
导游词之珠海轮廓
2019/10/25 职场文书
一道JS算法面试题——冒泡、选择排序
2021/04/21 Javascript
JavaScript文档对象模型DOM
2021/11/20 Javascript
Java工作中实用的代码优化技巧分享
2022/04/21 Java/Android