一、装饰器定义

  • 装饰器是一种特殊类型的声明,它能够被附加到类声明,方法,属性或参数上,可以修改类的行为
  • 通俗的讲装饰器就是一个方法,可以注入到类、方法、属性参数上来扩展类、属性、方法、参数的功能
  • 常见的装饰器有:类装饰器、属性装饰器、方法装饰器、参数装饰器
  • 装饰器的写法:普通装饰器(无法传参) 、 装饰器工厂(可传参)
  • 装饰器是过去几年中js最大的成就之一,已是Es7的标准特性之一

二、类装饰器

  • 类装饰器在类声明之前被声明(紧靠着类声明)。 类装饰器应用于类构造函数,可以用来监视,修改或替换类定义。 传入一个参数

1、类装饰器:普通装饰器(无法传参)
  1. function logClass(params: any) {
  2. console.log(params);
  3. // params 就是当前类 params === HttpClient
  4. params.prototype.apiUrl = '动态扩展的属性';
  5. params.prototype.run = function () {
  6. console.log('我是一个run方法');
  7. }
  8. }
  9. @logClass // 调用装饰器
  10. class HttpClient {
  11. constructor() {
  12. }
  13. getData() {
  14. }
  15. }
  16. var http: any = new HttpClient();
  17. console.log(http.apiUrl);
  18. http.run();

2、类装饰器:装饰器工厂(可传参)
function logClass(params: string) {
    return function (target: any) {
        console.log(target);
        console.log(params);
        target.prototype.apiUrl = params;
    }
}

@logClass('http://www.itying.com/api')
class HttpClient {
    constructor() {
    }
    getData() {
    }
}
var http: any = new HttpClient();
console.log(http.apiUrl);

3、下面是一个重载构造函数的例子
  • 类装饰器表达式会在运行时当作函数被调用,类的构造函数作为其唯一的参数
  • 如果类装饰器返回一个值,它会使用提供的构造函数来替换类的声明 ```typescript function logClass(target: any) { console.log(target); return class extends target {
      apiUrl: any = '我是修改后的数据';
      getData() {
          this.apiUrl = this.apiUrl + '----';
          console.log(this.apiUrl);
      }
    
    } }

@logClass class HttpClient { public apiUrl: string | undefined; constructor() { this.apiUrl = ‘我是构造函数里面的apiUrl’; } getData() { console.log(this.apiUrl); } } var http = new HttpClient(); http.getData();


<a name="2ddf62c5"></a>
### 三、属性装饰器

- 属性装饰器表达式会在运行时当作函数被调用,传入下列2个参数:
- 1、对于静态成员来说是类的构造函数,对于实例成员是类的原型对象
- 2、成员的名字
```typescript
// 属性装饰器
function logProperty(params: any) {
    // 属性装饰器表达式会在运行时当作函数被调用,传入下列2个参数
    return function (target: any, attr: any) {
        console.log(target);
        console.log(attr);
        target[attr] = params;
    }
}

@logClass('xxxx')
class HttpClient {
    @logProperty('http://www.baidu.com') // 属性装饰器,对url生效
    public url: any | undefined;
    constructor() {
    }
    getData() {
        console.log(this.url);
    }
}
var http = new HttpClient();
http.getData();