利用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 showModelDialog的使用方法示例详解
Nov 19 Javascript
红米手机抢购的js代码
Mar 10 Javascript
调试JavaScript中正则表达式中遇到的问题
Jan 27 Javascript
JavaScript AOP编程实例
Jun 16 Javascript
详解js运算符单竖杠“|”与“||”的用法和作用介绍
Nov 04 Javascript
完美解决UI-Grid表格元素中多个空格显示为一个空格的问题
Apr 25 Javascript
浅谈JS如何实现真正的对象常量
Jun 25 Javascript
sortable+element 实现表格行拖拽的方法示例
Jun 07 Javascript
微信小程序如何实现点击图片放大功能
Jan 21 Javascript
JS数组扁平化、去重、排序操作实例详解
Feb 24 Javascript
JS算法教程之字符串去重与字符串反转
Dec 15 Javascript
vue本地构建热更新卡顿的问题“75 advanced module optimization”完美解决方案
Aug 05 Vue.js
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编程中字符串处理的5个技巧小结
2007/11/13 PHP
PHP用SAX解析XML的实现代码与问题分析
2011/08/22 PHP
深入分析PHP引用(&amp;)
2014/09/04 PHP
PHP获取当前完整URL地址的函数
2014/12/21 PHP
PHP中判断文件存在使用is_file还是file_exists?
2015/04/03 PHP
WordPress中Gravatar头像缓存到本地及相关优化的技巧
2015/12/19 PHP
php实现博客,论坛图片防盗链的方法
2016/10/15 PHP
Ajax中的JSON格式与php传输过程全面解析
2017/11/14 PHP
PHP PDOStatement::errorInfo讲解
2019/01/31 PHP
Dom在ajax技术中的作用说明
2010/10/25 Javascript
Nodejs极简入门教程(二):定时器
2014/10/25 NodeJs
firefox浏览器用jquery.uploadify插件上传时报HTTP 302错误
2015/03/01 Javascript
JavaScript中逗号运算符介绍及使用示例
2015/03/13 Javascript
JavaScript获得表单target属性的方法
2015/04/02 Javascript
使用AngularJS创建单页应用的编程指引
2015/06/19 Javascript
详解PHP中pathinfo()函数导致的安全问题
2017/01/05 Javascript
JavaScript仿微信(电话)联系人列表滑动字母索引实例讲解(推荐)
2017/08/16 Javascript
浅谈vue-cli加载不到dev-server.js的解决办法
2017/11/24 Javascript
基于vue实现web端超大数据量表格的卡顿解决
2019/04/02 Javascript
jquery实现两个div中的元素相互拖动的方法分析
2020/04/05 jQuery
基于js判断浏览器是否支持webGL
2020/04/18 Javascript
Swift中的协议(protocol)学习教程
2016/07/08 Python
Python二叉树的定义及常用遍历算法分析
2017/11/24 Python
Python绘制KS曲线的实现方法
2018/08/13 Python
python机器学习包mlxtend的安装和配置详解
2019/08/21 Python
python爬虫 Pyppeteer使用方法解析
2019/09/28 Python
详解Python Opencv和PIL读取图像文件的差别
2019/12/27 Python
python实现定时发送邮件到指定邮箱
2020/12/23 Python
CSS3 input框的实现代码类似Google登录的动画效果
2020/08/04 HTML / CSS
英国山地公路自行车商店:Tweeks Cycles
2018/03/16 全球购物
自荐信范文
2013/12/10 职场文书
重阳节慰问信
2015/02/15 职场文书
小学生读书笔记
2015/07/01 职场文书
springboot临时文件存储目录配置方式
2021/07/01 Java/Android
详细聊聊关于Mysql联合查询的那些事儿
2021/10/24 MySQL
postgreSQL数据库基础知识介绍
2022/04/12 PostgreSQL