随着TypeScript和ES6里引入了类,在一些场景下我们需要额外的特性来支持标注或修改类及其成员。装饰器是一种特殊类型的声明,它能够被附加到类声明,方法,属性或参数上,可以修改类的行为。
通俗的讲,装饰器就是一个方法,可以注入到类、方法、属性、参数上来扩展类、方法、属性、参数的功能。
常见的装饰器有:普通装饰器(无法传参)、装饰器工厂(可传参)
装饰器工厂
如果我们要定制一个修饰器如何应用到一个声明上,我们得写一个装饰器工厂函数。 装饰器工厂就是一个简单的函数,它返回一个表达式,以供装饰器在运行时调用。
我们可以通过下面的方式来写一个装饰器工厂函数:
function color(value: string) { // 这是一个装饰器工厂
return function (target) { // 这是装饰器
// do something with "target" and "value"...
}
}
装饰器组合
多个装饰器可以同时应用到一个声明上,就像下面的示例:
- 书写在同一行上:
@f @g x
- 书写在多行上:
@f
@g
x
当多个装饰器应用于一个声明上,它们求值方式与复合函数相似。在这个模型下,当复合f和g时,复合的结果(f ∘ g)(x)等同于f(g(x))。下面的修饰器的执行顺序的例子中会举例说明。
装饰器类型 (五种装饰器)
- 类装饰器
- 属性装饰器
- 方法装饰器
- 函数参数装饰器
-
访问器装饰器
一、类装饰器
类装饰器在类声明之前被声明(紧靠着类声明)。 类装饰器应用于类构造函数,可以用来监视,修改或替换类定义。 类装饰器不能用在声明文件中( .d.ts
),也不能用在任何外部上下文中(比如declare
的类)。
类装饰器表达式会在运行时当作函数被调用,类的构造函数作为其唯一的参数。
如果类装饰器返回一个值,它会使用提供的构造函数来替换类的声明。
注意 如果你要返回一个新的构造函数,你必须注意处理好原来的原型链。 在运行时的装饰器调用逻辑中 不会为你做这些。
// 类装饰器(target 就是当前类)---- 普通装饰器(无法传参)
function logClass(target: any) {
console.log(target);
target.prototype.apiUrl = 'dfdjfadgadgadjfad';
target.prototype.run = function() {
console.log('这是扩展的run方法');
}
}
@logClass
class HttpClient {
constructor() {
}
getData() {
}
}
var http:any = new HttpClient();
console.log(http.apiUrl);
http.run();
// 类装饰器 ---- 装饰器工厂(可传参)
function logClass(params: string) {
console.log(params);
// target 就是当前类
return function(target: any) {
console.log(target);
console.log(params);
target.prototype.apiUrl = params;
}
}
@logClass('http://www.baidu.com')
class HttpClient {
constructor() {