详解es6超好用的语法糖Decorator


Posted in Javascript onAugust 01, 2018

Decorator(修饰器/装饰器)是es6提出的语法糖,用于修改类的行为。不过目前主流浏览器都没有很好的支持,我们需要用babel来转换为浏览器能识别的语言。在这篇文章中将介绍decorator的基础用法和一些应用实例。

1.修饰类

(1) 基础用法

@testable
class MyClass{}

function testable(target){
  target.isTestable=true
}

console.log(MyClass.isTestable) // true

贴一下babel转换后的代码,

var _class;

let MyClass = testable(_class = class MyClass {}) || _class;

function testable(target) {
  target.isTestable = true;
}

也可以在prototype上修改属性,也可以为修饰器添加多个参数。

@testable(false)
class MyAnotherClass{
  
}
function testable(status){
  return target=>{target.prototype.isTestable=status}
}
console.log('MyClass.isTestable',MyAnotherClass.prototype.isTestable) // false

当然我们通过修饰器,把某个对象的方法添加到目标类的实例上,注意要在类的prototype上添加。

const foo={isTestable:true}
function testable(...list){
  return target=>{Object.assign(target.prototype,...list)}
}

@testable(foo)
class MyAnotherClass{}
const obj=new MyAnotherClass()

console.log('MyClass.isTestable',obj.isTestable) // true

(2) 应用

React App的开发中,使用redux通常需要react-redux中的connect方法,将两者结合在一起。通常的写法是:

class MyReactComponent extends React.Component {}

export default connect(mapStateToProps, mapDispatchToProps)(MyReactComponent);

如果使用decorator,代码可读性更高了一些。

@connect(mapStateToProps, mapDispatchToProps)
export default class MyReactComponent extends React.Component {}

2.修饰方法

(1).基础用法

// target:在方法中的target指向类的prototype
function readonly(target,key,descriptor){
  descriptor.writable=false
  return descriptor
}

class MyClass{
  @readonly
  print(){console.log(`a:${this.a}`)}
}

(2).js中Object的属性

var person = {}
Object.defineProperty(person,'name',{
  configurable:false,//能否使用delete、能否需改属性特性、或能否修改访问器属性、,false为不可重新定义,默认值为true
  enumerable:false,//对象属性是否可通过for-in循环,flase为不可循环,默认值为true
  writable:false,//对象属性是否可修改,flase为不可修改,默认值为true
  value:'xiaoming' //对象属性的默认值,默认值为undefined
});

对应到descriptor为下面四个属性:

{
   value: specifiedFunction,
   enumerable: false,
   configurable: true,
   writable: true
};

(3). 应用

我们开始写一个@log修饰器,可以输出日志:

class Math{
  @log
  add(a,b){
    return a+b
  }
}

const math=new Math()
math.add(1,2)

function log(target,name,descriptor){
  const oldValue=descriptor.value

  descriptor.value=function(){
    console.log(`calling ${name} with ${JSON.stringify(arguments)}`)
    return oldValue.apply(this,arguments)
  }

  return descriptor
}

上面的代码中,@log作用是在返回结果前,打印函数名和其参数,起到输出日至的作用。上面的程序运行后,控制台将输出:

calling add with {"0":1,"1":2}

(4). 多个修饰器

良好命名的修饰器可以起到简洁注释的作用,如下:

class Example {
  @readonly
  @enumable
  method(){}
}

多个修饰器的执行顺序是由外向内进入;再由内向外执行。

class Example {
  @decorator(1)
  @decorator(2)
  method(){}
}

function decorator(id){
  console.log('id is ',id)
  return (target,property,descriptor)=>console.log('executed',id)
}

控制台输出

id is  1
id is  2
executed 2
executed 1

附录:babel配置

babel插件transform-decorators还没有正式版,我们可以用transform-decorators-legacy

安装babel

yarn add babel-plugin-transform-decorators-legacy babel-preset-es2017

配置.babelrc

{
  "presets": ["es2017"],
  "plugins":[
    "transform-decorators-legacy"
  ]
}

执行编译后的文件

因为我们为了测试,没必要非得放在浏览器里看了,可以用node执行babel转换后的文件。直接运行yarn start

// package.json

 "scripts": {
  "build": "babel ./decorator -d lib",
  "start":"yarn build && node ./lib/index.js"
 },

参考链接

ECMAScript 6 入门 -- 修饰器

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
JS控件autocomplete 0.11演示及下载 1月5日已更新
Jan 09 Javascript
jquery中常用的SET和GET
Jan 13 Javascript
js 页面执行时间计算代码
Mar 04 Javascript
Javascript 面向对象 对象(Object)
May 13 Javascript
页面调用单个swf文件,嵌套出多个方法。
Nov 21 Javascript
快速解决jquery之get缓存问题的最简单方法介绍
Dec 19 Javascript
js实现浏览本地文件并显示扩展名的方法
Aug 17 Javascript
bootstrap suggest下拉框使用详解
Apr 10 Javascript
vue组件初学_弹射小球(实例讲解)
Sep 06 Javascript
jquery 给动态生成的标签绑定事件的几种方法总结
Feb 24 jQuery
vue生命周期的探索
Apr 03 Javascript
通过vue刷新左侧菜单栏操作
Aug 06 Javascript
Vue Router去掉url中默认的锚点#
Aug 01 #Javascript
vue定义全局变量和全局方法的方法示例
Aug 01 #Javascript
node.js遍历目录的方法示例
Aug 01 #Javascript
深入浅出理解JavaScript闭包的功能与用法
Aug 01 #Javascript
Angular路由ui-router配置详解
Aug 01 #Javascript
javascript数据结构之多叉树经典操作示例【创建、添加、遍历、移除等】
Aug 01 #Javascript
JavaScript事件冒泡与事件捕获实例分析
Aug 01 #Javascript
You might like
百事可乐也出咖啡了 双倍咖啡因双倍快乐
2021/03/03 咖啡文化
PHP 时间转换Unix时间戳代码
2010/01/22 PHP
php实现源代码加密的方法
2015/07/11 PHP
Zend Framework自定义Helper类相关注意事项总结
2016/03/14 PHP
Yii中CGridView禁止列排序的设置方法
2016/07/12 PHP
Yii2增删改查之查询 where参数详细介绍
2016/08/08 PHP
jquery设置控件位置的方法
2013/08/21 Javascript
jquery 操作iframe的几种方法总结
2013/12/13 Javascript
取消选中单选框radio的三种方式示例介绍
2013/12/23 Javascript
json与jsonp知识小结(推荐)
2016/08/16 Javascript
Json按某个键的值进行排序
2016/12/22 Javascript
深入理解vue-router之keep-alive
2017/08/31 Javascript
详解javascript常用工具类的封装
2018/01/30 Javascript
JavaScript数组去重算法实例小结
2018/05/07 Javascript
node全局变量__dirname与__filename的区别
2019/01/14 Javascript
vue 实现小程序或商品秒杀倒计时
2019/04/14 Javascript
解决vue 单文件组件中样式加载问题
2019/04/24 Javascript
react高阶组件添加和删除props
2019/04/26 Javascript
element的el-table中记录滚动条位置的示例代码
2019/11/06 Javascript
基于vue与element实现创建试卷相关功能(实例代码)
2020/12/07 Vue.js
python抓取京东商城手机列表url实例代码
2013/12/18 Python
python爬虫获取多页天涯帖子
2018/02/23 Python
flask-socketio实现WebSocket的方法
2018/07/31 Python
python中利用numpy.array()实现俩个数值列表的对应相加方法
2019/08/26 Python
python安装本地whl的实例步骤
2019/10/12 Python
Python3使用xlrd、xlwt处理Excel方法数据
2020/02/28 Python
优秀毕业生求职信范文
2014/01/02 职场文书
自荐信需注意事项
2014/01/25 职场文书
施工材料员岗位职责
2014/02/12 职场文书
公司募捐倡议书
2014/05/14 职场文书
学校综治宣传月活动总结
2014/07/02 职场文书
电子商务专业求职信
2014/07/10 职场文书
计算机应用应届生求职信
2014/07/12 职场文书
政府个人对照检查材料
2014/08/28 职场文书
新年寄语2016
2015/08/17 职场文书
MySQL root密码的重置方法
2021/04/21 MySQL