2

我可以使用创建自定义装饰器reflect-metadata,它工作正常。

问题是,我不知道如何获取所有实例装饰器。

import 'reflect-metadata';

console.clear();

function readTypes() {
  const decorator: MethodDecorator = (target, propertyKey, description) => {
    const args = Reflect.getMetadata(
      'design:paramtypes',
      target,
      propertyKey
    ).map(c => c.name);
    const ret = Reflect.getMetadata('design:returntype', target, propertyKey);
    console.log(`Arguments type: ${args.join(', ')}.`);
    console.log(`Return type:    ${ret.name}.`);
  };
  return decorator;
}

class Foo {}

class Bar {
  @readTypes()
  public fn(a: number, b: string, c: Foo): boolean {
    return true;
  }
}

const barInstance = new Bar();

我想@readTypesbarInstance. 我该怎么做?

请参阅工作示例: https ://stackblitz.com/edit/decorators-metadata-example-nakg4c

4

1 回答 1

1

首先,您没有编写任何元数据,而只是读取它。如果您想查找哪些属性被修饰,那么您必须将元数据写入这些属性。

为简单起见,让我们将装饰器简化为:

// It's a best practice to use symbol as metadata keys.
const isReadTypesProp = Symbol('isReadTypesProp')

function readTypes() {
  const decorator: MethodDecorator = (target, propertyKey, description) => {
    Reflect.defineMetadata(isReadTypesProp, true, target, propertyKey);
  };
  return decorator;
}

现在,当使用装饰器时,它会在解析类时执行,而不是在创建实例时执行。这意味着target装饰器中的函数实际上是prototype类的构造函数。

换句话说,是与or或target相同的对象 。Bar.prototypebarInstance.constructor.prototypebarInstance.__proto__

知道我们可以遍历原型对象中的所有属性名称并查找我们之前设置的元数据:

function getReadTypesPropsFromInstance(object: {
  constructor: {
    prototype: unknown;
  };
}) {
  const target = object.constructor.prototype;
  const keys = Object.getOwnPropertyNames(target);
  return keys.filter(key => Reflect.getMetadata(isReadTypesProp, target, key));
}

现在返回装饰属性的名称:

const barInstance = new Bar();
console.log(getReadTypesPropsFromInstance(barInstance)); // ["fn"]

工作示例

于 2021-07-22T22:41:17.193 回答