深入理解和应用JavaScript中的装饰器

JavaScript一直在发展和吸取新的附加功能,以增强其对开发者的吸引力。其中一个新语法就是装饰器。装饰器对于开发者来说,是一枚强大的工具,它可以在忽略函数的内部巧妙地修改其行为。

什么是装饰器?

装饰器就是一个函数,它可以修改类的行为或属性,而不需要知道类内部的具体实现。装饰器用于提高代码的可读性和健壮性,同时缩减冗余的代码。

如何使用装饰器?

让我们首先查看一下如何在JS中使用装饰器。在编写装饰器时,你需要将要装饰的对象,其原始描述符和装饰器的目标传给装饰器。

例如,我们有一个讲述狗叫声的函数,我们想要装饰它:

scss 复制代码
function dogBark() {
  console.log('Woof woof');
}

dogBark();

这是一段简单的代码,但我们希望这个函数在每次调用时都打印出函数被调用的时间。这就是我们使用装饰器的地方。首先我们需要创建一个装饰器:

javascript 复制代码
function decorator(func) {
  return function () {
    console.log(`Function ${func.name} was called at: ${new Date().toLocaleTimeString()}`);
    func();
  };
}

然后我们把这个装饰器应用到我们的函数上:

ini 复制代码
dogBark = decorator(dogBark);
dogBark();

现在,每次我们调用dogBark函数时,都会打协议输出函数被调用的时间。

类装饰器

到此为止,我们已经成功地创建了一个将功能添加到单个函数的装饰器。那么如果我们想要添加到类上呢?

让我们设置一个类,并为其添加一些装饰:

typescript 复制代码
@logger
class Dog {
  constructor() {
    console.log('new dog was created');
  }
}

在这段代码中,我们使用 "@logger" 装饰器装饰了整个 "Dog" 类。那么问题来了,这个 "@logger" 装饰器是什么样的呢?这个装饰器需要注入到类的原型链中,以此来拦截类的实例。

下面,我们就来看看这个装饰器:

javascript 复制代码
function logger(target) {
  let original = target;

  function construct(constructor, args) {
    let instance = new Function('origin', 'return new origin(...args)')(original, args);
    console.log(`New dog was created at: ${new Date().toLocaleTimeString()}`);
    return instance;
  }

  let prototype = Object.create(original.prototype);
  let surrogate = function (...args) { return construct(original, args); };
  surrogate.prototype = prototype;
  return surrogate;
}

这个装饰器有点复杂,但仔细分析后你会发现,它在实例化类之后,打印出一条消息显示实例创建的时间。

属性和方法装饰器

最后,我们可以通过装饰器来修改类的属性和方法。

例如,我们有这样一个类:

javascript 复制代码
class Dog {
  bark() {
    console.log('Woof woof');
  }

  @uppercase
  breed = 'Poodle';
}

function uppercase(target, key) {
  let value = target[key];

  if (delete target[key]) {
    Object.defineProperty(target, key, {
      get: () => value,
      set: (newVal) => value = newVal.toUpperCase(),
      enumerable: true,
      configurable: true
    });
  }
}

在这份代码中,我们创建了一个装饰器将狗的品种强制大写。

结语

JS装饰器是一个强大的工具,允许我们在不修改原始函数或类的情况下增加新功能。装饰器在很多流行的JS库中都有使用,掌握装饰器也许会为你的JS生涯开辟新的可能性。

相关推荐
博一波24 分钟前
【设计模式-行为型】解释器模式
设计模式·解释器模式
爱上大树的小猪28 分钟前
【前端】Electron入门开发教程,从介绍Electron到基础引用以及部分深度使用,附带常见的十个报错问题的解决方案和代码优化。
前端·javascript·electron
福大大架构师每日一题29 分钟前
2.6 createCmd中的builder建造者设计模式
设计模式·kubernetes
呦呦鹿鸣Rzh1 小时前
实现标题-超链接
java·前端·javascript
Super毛毛穗1 小时前
GeoJSON 数据
javascript·gis·geojson
Tester_孙大壮1 小时前
第31章 测试驱动开发中的设计模式与重构解析(Python 版)
python·设计模式·重构
网络点点滴2 小时前
声明式和函数式 JavaScript 原则
开发语言·前端·javascript
禁默2 小时前
【学术会议-第五届机械设计与仿真国际学术会议(MDS 2025) 】前端开发:技术与艺术的完美融合
前端·论文·学术
纯粹的摆烂狗2 小时前
深圳大学-智能网络与计算-实验四:云-边协同计算实验
javascript
binnnngo2 小时前
2.体验vue
前端·javascript·vue.js