利用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[js]获取url参数的代码
Oct 17 Javascript
如何使Chrome控制台支持多行js模式——意外发现
Jun 13 Javascript
JS 实现BASE64_ENCODE和BASE64_DECODE(实例代码)
Nov 13 Javascript
初识SmartJS - AOP三剑客
Jun 08 Javascript
javascript如何实现360度全景照片问题汇总
Apr 04 Javascript
js带闹铃功能的倒计时代码
Sep 29 Javascript
Bootstrap模态框(Modal)实现过渡效果
Mar 17 Javascript
详解如何在React组件“外”使用父组件的Props
Jan 12 Javascript
对vue里函数的调用顺序介绍
Mar 17 Javascript
ES6模板字符串和标签模板的应用实例分析
Jun 25 Javascript
mpvue微信小程序的接口请求fly全局拦截代码实例
Nov 13 Javascript
node.js中npm包管理工具用法分析
Feb 14 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
生成ubuntu自动切换壁纸xml文件的php代码
2010/07/17 PHP
PHP自动重命名文件实现方法
2014/11/04 PHP
最准确的php截取字符串长度函数
2015/10/29 PHP
PHP实现文件上传和多文件上传
2015/12/24 PHP
PHP之十六个魔术方法详细介绍
2016/11/01 PHP
实例讲解PHP验证邮箱是否合格
2019/01/28 PHP
php实现多站点共用session实现单点登录的方法详解
2019/09/18 PHP
jQuery 名称冲突的解决方法
2011/04/08 Javascript
js复制到剪切板的实例方法
2013/06/28 Javascript
js获取input标签的输入值实现代码
2013/08/05 Javascript
Jquery validation remote 验证的缓存问题解决方法
2014/03/25 Javascript
javascript实现复选框超过限制即弹出警告框的方法
2015/02/25 Javascript
JS实现文字链接感应鼠标淡入淡出改变颜色的方法
2015/02/26 Javascript
使用vue.js实现checkbox的全选和多个的删除功能
2017/02/17 Javascript
微信小程序开发教程之增加mixin扩展
2017/08/09 Javascript
全选复选框JavaScript编写小结(附代码)
2017/08/16 Javascript
Vue-cli创建项目从单页面到多页面的方法
2017/09/20 Javascript
vue.js引入外部CSS样式和外部JS文件的方法
2019/01/06 Javascript
12 种使用Vue 的最佳做法
2020/03/30 Javascript
小程序自定义圆形进度条
2020/11/17 Javascript
玩转python爬虫之爬取糗事百科段子
2016/02/17 Python
Python优化技巧之利用ctypes提高执行速度
2016/09/11 Python
python3.6.3转化为win-exe文件发布的方法
2018/10/31 Python
解决python ogr shp字段写入中文乱码的问题
2018/12/31 Python
我用Python抓取了7000 多本电子书案例详解
2019/03/25 Python
django写用户登录判定并跳转制定页面的实例
2019/08/21 Python
Django 限制访问频率的思路详解
2019/12/24 Python
CSS3绘制不规则图形的一些方法示例
2015/11/07 HTML / CSS
Jimmy Choo美国官网:周仰杰鞋子品牌
2018/06/08 全球购物
高三英语教学反思
2014/01/13 职场文书
高二美术教学反思
2014/01/14 职场文书
护理专业大学生自我推荐信
2014/01/25 职场文书
环保标语口号
2014/06/13 职场文书
超市主管竞聘书
2015/09/15 职场文书
html5 录制mp3音频支持采样率和比特率设置
2021/07/15 Javascript