likes
comments
collection
share

一起探索ES7修饰器的最新提案

作者站长头像
站长
· 阅读数 61

Node环境中使用修饰器

node: v14.16.0

安装 babel 相关的包

"devDependencies": {
  "@babel/core": "^7.22.5",
  "@babel/node": "^7.22.5",
  "@babel/plugin-proposal-decorators": "^7.22.5",
  "@babel/preset-env": "^7.22.5"
}

配置 .babelrc

{
  "presets": ["@babel/preset-env"],
  "plugins": [
    ["@babel/plugin-proposal-decorators", { "version": "2023-01" }]
  ]
}

www.babeljs.cn/docs/babel-…

执行代码

npx babel-node index.js

注意事项

在开始编写修饰器之前,我们要格外注意一个配置,那就是修饰器的 版本 versions2023-01

// .babelrc
["@babel/plugin-proposal-decorators", { "version": "2023-01" }]

我们这里使用的是 @babel/plugin-proposal-decorators 最新提案,如果你使用之前的提案,如 version: legacy,那么在使用修饰器时,结果会有很大的差别。

调用装饰器

当装饰器被调用时,它们接收两个参数:

  1. 被修饰的对象(class, 成员方法)

  2. 包含有关被装饰值的信息的上下文对象

使用 TypeScript 接口来描述的话,就是如下样子:

type Decorator = (value: Input, context: {
  kind: string; // "class" | "method" | "getter" | "setter" | "field" | "accessor"
  name: string | symbol; // 值的名称,或者在私有元素的情况下是它的描述(例如可读名称)
  access: { // 包含访问值的方法的对象
    get?(): unknown;
    set?(value: unknown): void;
  };
  private?: boolean; // 该值是否为私有类元素。仅适用于类元素。
  static?: boolean; // 该值是否为static类元素。仅适用于类元素
  addInitializer?(initializer: () => void): void; // 允许用户添加额外的初始化逻辑。
}) => Output | void;

Input、Output 代表传递给装饰器的对象(修饰的对象)和从装饰器返回的值。

装饰器就是调用了普通函数,定义装饰器没有特殊的语法,任何函数都可以用作装饰器。

修饰类

我们编写一个修饰类的修饰器 mixins,目的是将多个对象,混入到类的原型中。

定义类

class Person {
  name = 'alex.cheng'
}

接着定义 mixins 修饰器,并在 Person 类上使用它,foo bar 是我们要添加到 Person 原型上的对象。

const foo = {
  a: 1
}

const bar = {
  b: 2
}

function mixins(...list) {
  return function(value, context) {
    if (context.kind === 'class') {
      Object.assign(value.prototype, ...list)
    }
  }
}

class Person {
  name = 'alex.cheng'
}

就这样,我们将 foo 和 bar 对象通过 mixins 修饰器,添加到了 Person 的原型对象上。

我们就可以在每个 Person 实例上访问这两个对象了。

const p = new Person()

console.log(p.a, p.b)

如果我们不给修饰器传递参数,那么直接将 mixins 写成如下形式即可:

function mixins(value, context) {
  value.prototype.bar = {
    value: 'is bar'
  }
}

修饰成员方法

class Person {
  name = 'alex.cheng'

  @decoratorfn
  sayName() {
    console.log(this.name)
  }
}

function decoratorfn(value, context) {
  if (context.kind === 'method') {
    return function(...args) {
      const res = value.call(this, ...args)
    } 
  }
}

修饰器修饰方法时,可以粗略的转译为下面这种形式

class Person {
  sayName(arg) {}
}

Person.prototype.sayName = decoratorfn(Person.prototype.sayName, {
  kind: "method",
  name: "sayName",
  static: false,
  private: false,
}) ?? Person.prototype.sayName;

多个修饰器时,执行顺序是怎样的?

function first(...args) {
  console.log('first outer')
  return function(value, context) {
    console.log('first inner')
  }
}

function decoratorfn(...args) {
  console.log('decoratorfn outer')
  return function(value, context) {
    console.log('decoratorfn inner')
  }
}

class Person {
  name = 'alex.cheng'

  @first(1,2,3)
  @decoratorfn(4, 5, 6)
  sayName() {
    console.log(this.name)
  }
}

从上往下,从内到外

执行结果如下

first outer

decoratorfn outer

decoratorfn inner

first inner

修饰类的访问器属性

用 TypeScript 来描述访问器的修饰器方法,如下所示

type ClassGetterDecorator = (value: Function, context: {
  kind: "getter";
  name: string | symbol;
  access: { get(): unknown };
  static: boolean;
  private: boolean;
  addInitializer(initializer: () => void): void;
}) => Function | void;

type ClassSetterDecorator = (value: Function, context: {
  kind: "setter";
  name: string | symbol;
  access: { set(value: unknown): void };
  static: boolean;
  private: boolean;
  addInitializer(initializer: () => void): void;
}) => Function | void;

访问器也是方法,所以我们可以按照修饰方法时的处理逻辑来就可以了。

function getterDecorator(value, context) {
  if (context.kind === 'getter' || context.kind === 'setter') {
    // ...
  }
}

class Person {
  name = 'alex.cheng'

  @getterDecorator
  get name() {
    console.log('??', this.name)
  }

  set name(value) {
    this.name = value
  }
}

目前尚不清楚哪些用例受益于 getter/setter 合并。移除 getter/setter 合并是规范的一大简化,我们希望它也能简化实现。

修饰类的成员属性

type ClassFieldDecorator = (value: undefined, context: {
  kind: "field";
  name: string | symbol;
  access: { get(): unknown, set(value: unknown): void };
  static: boolean;
  private: boolean;
}) => (initialValue: unknown) => unknown | void;

注意,修饰属性的时候,第一个参数是 undefined,context.kind 是 field

function propDecorator(value, context) {
  return function(initialValue) {
    return initialValue + ' emmmmmm'
  }
}
class Person {
  @propDecorator name = 'alex.cheng'
}

实例化 Person

const p = new Person()

console.log(p.name) // alex.cheng emmmmmm

class 自动访问器

我们先看下定义 class 私有属性 #

class Person {
  #name = 'alex.cheng'
  accessor x = 1

  sayName() {
    console.log('#name', this.#name)
  }

  setName(value) {
    this.#name = value
  }
}


const p = new Person()

p.setName('hello world')

console.log(p.name) // undefined

p.sayName() // hello world

现在,我们给某个字段加上 自动访问器

class Person {
  accessor name = 'alex.cheng';
}

脱糖后,大概是如下样子

class Person {
  #name = 1;

  get name() {
    return this.#name;
  }

  set name(val) {
    this.#name = val;
  }
}

也可以定义静态和私有自动访问器:

class Person {
  static accessor age = 18;
  accessor #name = 'alex.cheng';
}

Person.age // 18

与字段装饰器不同,自动访问器装饰器接收一个值,该值是一个包含在类原型上定义的访问器的对象(或者在静态自动访问器的情况下是类本身)get、set。

然后装饰器可以包装这些并返回一个新的 get和/或set,允许装饰器拦截对属性的访问。

这是字段无法实现的功能,但自动访问器可以实现。

此外,自动访问器可以返回一个init函数,可用于更改私有属性的初始值,类似于字段装饰器。如果返回一个对象但省略了任何值,则省略值的默认行为是使用原始行为。如果返回包含这些属性的对象以外的任何其他类型的值,则会抛出错误。

function logged(value, { kind, name }) {
  if (kind === "accessor") {
    let { get, set } = value;

    return {
      get() {
        console.log(`getting ${name}`);

        return get.call(this);
      },

      set(val) {
        console.log(`setting ${name} to ${val}`);

        return set.call(this, val);
      },

      init(initialValue) {
        console.log(`initializing ${name} with value ${initialValue}`);
        return initialValue;
      }
    };
  }

  // ...
}

class C {
  @logged accessor x = 1;
}

let c = new C();
// initializing x with value 1
c.x;
// getting x
c.x = 123;
// setting x to 123

添加初始化逻辑addInitializer

该addInitializer方法在提供给装饰器的上下文对象上可用,用于除类字段之外的每种类型的值。可以调用此方法将初始化函数与类或类元素相关联,该函数可用于在定义值后运行任意代码以完成设置。这些初始化器的时间取决于装饰器的类型:

  • 类装饰器初始值设定项在类已完全定义后运行,并在类静态字段已分配后运行。
  • 类元素初始值设定项在类构造期间运行,在类字段初始化之前。
  • 类静态元素初始值设定项在类定义期间运行,在定义静态类字段之前,但在定义类元素之后。

我们可以使用addInitializer类装饰器来创建一个在浏览器中注册 Web 组件的装饰器。

function customElement(name) {
  return (value, { addInitializer }) => {
    addInitializer(function() {
      customElements.define(name, this);
    });
  }
}

@customElement('my-element')
class MyElement extends HTMLElement {
  static get observedAttributes() {
    return ['some', 'attrs'];
  }
}

这个例子粗略地“脱糖”到以下内容(即,可以这样转译):

class MyElement {
  static get observedAttributes() {
    return ['some', 'attrs'];
  }
}

let initializersForMyElement = [];

MyElement = customElement('my-element')(MyElement, {
  kind: "class",
  name: "MyElement",
  addInitializer(fn) {
    initializersForMyElement.push(fn);
  },
}) ?? MyElement;

for (let initializer of initializersForMyElement) {
  initializer.call(MyElement);
}

我们还可以使用addInitializerwith 方法装饰器来创建一个@bound装饰器,它将方法绑定到类的实例:

function bound(value, { name, addInitializer }) {
  addInitializer(function () {
    this[name] = this[name].bind(this);
  });
}

class C {
  message = "hello!";

  @bound
  m() {
    console.log(this.message);
  }
}

let { m } = new C();

m(); // hello!

这个例子大致对以下内容“脱糖”:

class C {
  constructor() {
    for (let initializer of initializersForM) {
      initializer.call(this);
    }

    this.message = "hello!";
  }

  m() {}
}

let initializersForM = []

C.prototype.m = bound(
  C.prototype.m,
  {
    kind: "method",
    name: "m",
    static: false,
    private: false,
    addInitializer(fn) {
      initializersForM.push(fn);
    },
  }
) ?? C.prototype.m;

一起交流

不管你遇到什么问题,或者是想交个朋友一起探讨技术(=。=),都可以加入我们,和我们一起探索未来 ~

喜欢这部分内容,就加入我们的QQ群,和大家一起交流技术吧~

QQ群1032965518

转载自:https://juejin.cn/post/7243019785100509243
评论
请登录