Typescript 装饰器混乱

问题描述 投票:0回答:1

我一直在使用 TypeScript 装饰器,但只是没有运气让它们工作。我阅读了如何实现打字稿装饰器?以及http://blog.wolksoftware.com/decorators-reflection-javascript-typescript,我从中创建了装饰器实现

function log(target: any, key: string, descriptor?: any) {

  // save a reference to the original method
  // this way we keep the values currently in the 
  // descriptor and don't overwrite what another 
  // decorator might have done to the descriptor.
  if(descriptor === undefined) {
      descriptor = Object.getOwnPropertyDescriptor(target, key);
  }
  var originalMethod = descriptor.value; 

  //editing the descriptor/value parameter
  descriptor.value =  function (...args: any[]) {
      var a = args.map(a => JSON.stringify(a)).join();
      // note usage of originalMethod here
      var result = 12;//originalMethod.apply(this, args);
      var r = JSON.stringify(result);
      console.log(`Call: ${key}(${a}) => ${r}`);
      return result;
  }

  // return edited descriptor as opposed to overwriting 
  // the descriptor by returning a new descriptor
  return descriptor;
}

class C {
    constructor(){
        console.log("built");
    }
    @log
    public foo(n: number) {
        console.log("foo called");
        return n * 2;
    }
}
//new MethodDecoratorExample().method("hi");
var c = new C();
var r = c.foo(23); //  "Call: foo(23) => 12"
console.log(r);

但是运行这段代码并没有产生我所期望的结果,事实上它似乎没有覆盖描述符。深入研究后,我发现在生成的代码中对

__decorate
的调用只有三个参数

__decorate([
    log
], C.prototype, "foo");

这意味着在 __decorate c = 3 中,描述符没有传入。因此,新的描述符没有返回,代码流继续进行,就好像该方法没有被拦截一样。

所以我一定是错误地应用了这个装饰器,但我看不出我是如何做错的。它可能被解释为属性描述符吗?我确实在某处看到了一些关于此的评论,但这与将 C 中的 foo 方法定义为 lambda 相关,但我没有这样做。

我怎样才能让它工作?

typescript typescript-decorator
1个回答
1
投票

这里的问题是 TypeScript 默认会生成 ES3 代码,而为 ES3 生成的方法装饰器显然与为 ES5 生成的方法有些不同。因此,除非您确实需要以 ES3 为目标,否则解决问题的简单方法是使用

tsc --experimentalDecorators --target es5
来针对 ES5,或使用
tsc --experimentalDecorators --target es6
来针对 ES6。

© www.soinside.com 2019 - 2024. All rights reserved.