JS 装饰器模式实战 – 使用 Decorators 增强类与方法的优雅方案

装饰器模式通过@语法为类和方法非侵入式添加功能,如日志、权限、性能监控等横切关注点,提升代码复用性与可维护性。

js 装饰器模式实战 - 使用 decorators 增强类与方法的优雅方案

JavaScript 装饰器模式,说白了,就是一种非常优雅、声明式地增强类和方法功能的方式。它允许你在不修改原有代码结构的前提下,为它们“附加”新的行为或元数据。在我看来,这就像给你的代码穿上了一件件定制的“外套”,让它们在保持核心功能不变的同时,拥有了更多酷炫的能力。这套方案的核心价值在于解耦和复用,让那些原本散落在各处的横切关注点(比如日志、权限、性能监控)能够集中管理,大大提升了代码的可读性和可维护性。

解决方案

装饰器(Decorators)本质上就是一种特殊类型的函数,它能够修改或替换类、方法、访问器、属性或参数的定义。在 JavaScript 中,我们通常通过

@

符号来使用它,就像这样:

@decoratorName

。这种语法糖让代码看起来非常直观,一眼就能看出某个类或方法被“装饰”了什么功能。

它的工作原理是,当你定义一个类或方法时,装饰器会在它们被定义时立即执行。它会接收到被装饰的目标(比如类构造函数、方法的描述符等),然后返回一个修改后的目标,或者干脆返回一个新的目标来替换原来的。这种“运行时修改定义”的能力,使得我们可以在不侵入原有业务逻辑的情况下,注入各种辅助功能。

举个最简单的例子,如果你想给一个方法加上日志功能,传统做法可能是在方法内部的开头和结尾都写上

console.log

。但如果有很多方法都需要日志呢?代码就会变得很冗余。而有了装饰器,你只需要写一个

@log

装饰器,然后把它放在任何需要日志的方法上面,瞬间就解决了重复劳动的问题。这种声明式的用法,让代码意图更加清晰,也更容易维护。

// 假设这是我们的日志装饰器function log(target, propertyKey, descriptor) {  const originalMethod = descriptor.value;  descriptor.value = function (...args) {    console.log(`[LOG] Calling method: ${propertyKey.toString()} with args: ${JSON.stringify(args)}`);    const result = originalMethod.apply(this, args);    console.log(`[LOG] Method ${propertyKey.toString()} returned: ${JSON.stringify(result)}`);    return result;  };  return descriptor;}class Calculator {  @log  add(a, b) {    return a + b;  }  @log  subtract(a, b) {    return a - b;  }}const calc = new Calculator();calc.add(2, 3);// [LOG] Calling method: add with args: [2,3]// [LOG] Method add returned: 5calc.subtract(5, 1);// [LOG] Calling method: subtract with args: [5,1]// [LOG] Method subtract returned: 4

可以看到,

@log

装饰器在不改变

add

subtract

方法内部逻辑的情况下,为它们添加了日志功能。这正是装饰器模式的魅力所在。

装饰器在前端开发中,具体能解决哪些令人头疼的重复性问题?

说实话,在日常前端开发中,我们经常会遇到一些横切关注点,它们本身不是核心业务逻辑,却又无处不在,比如数据校验、权限控制、性能统计、事件绑定等等。这些东西如果每次都手写一遍,或者通过继承、组合的方式去处理,代码会变得非常臃肿,而且难以维护。装饰器在这方面简直是“神来之笔”,它能把这些重复性工作抽离出来,以一种非常优雅的方式注入到目标代码中。

1. 性能监控与埋点:假设你想知道一个方法执行了多久,或者某个组件渲染了多少次。传统做法可能是在方法开始前记录时间,结束后计算差值。但有了装饰器,你可以创建一个

@measurePerformance

装饰器。把它加到任何你关心性能的方法上,它就能自动帮你统计并输出执行时间。这对于优化性能瓶颈,或者做一些用户行为分析的埋点,简直是太方便了。

function measurePerformance(target, propertyKey, descriptor) {  const originalMethod = descriptor.value;  descriptor.value = function (...args) {    const start = performance.now();    const result = originalMethod.apply(this, args);    const end = performance.now();    console.log(`[Performance] Method ${propertyKey.toString()} executed in ${end - start}ms.`);    return result;  };  return descriptor;}class DataProcessor {  @measurePerformance  processLargeDataSet(data) {    // 模拟耗时操作    let sum = 0;    for (let i = 0; i < 1000000; i++) {      sum += i;    }    return data.length + sum;  }}const processor = new DataProcessor();processor.processLargeDataSet([1, 2, 3]);// [Performance] Method processLargeDataSet executed in XXms.

2. 权限控制与认证:在许多应用中,某些操作需要特定的用户权限。你不可能在每个方法里都写一遍

if (!user.hasPermission('admin')) return;

。这简直是灾难。这时候,一个

@requiresRole('admin')

@isAuthenticated

装饰器就能派上大用场。它可以在方法执行前检查用户的权限,如果权限不足,就直接阻止方法执行或抛出错误。这大大简化了权限逻辑的实现和管理。

function requiresRole(role) {  return function (target, propertyKey, descriptor) {    const originalMethod = descriptor.value;    descriptor.value = function (...args) {      // 假设这里有一个全局的用户信息或权限检查服务      // 实际应用中会从JWT、Session或Redux Store中获取      const currentUser = { roles: ['user'] }; // 模拟当前用户角色      if (!currentUser.roles.includes(role)) {        console.warn(`[Auth] Access denied for ${propertyKey.toString()}. Required role: ${role}`);        throw new Error(`Permission denied: requires ${role} role.`);      }      return originalMethod.apply(this, args);    };    return descriptor;  };}class AdminPanel {  @requiresRole('admin')  deleteUser(userId) {    console.log(`Deleting user: ${userId}`);    // ... 执行删除操作  }  @requiresRole('user') // 即使是普通用户也可以访问  viewDashboard() {    console.log('Viewing dashboard.');  }}const adminPanel = new AdminPanel();try {  adminPanel.deleteUser(123); // 会抛出权限不足的错误} catch (e) {  console.error(e.message);}adminPanel.viewDashboard(); // 正常执行

3. 表单验证与数据处理:在处理用户输入时,验证是必不可少的一环。你可以创建

@validate(schema)

装饰器,在方法接收参数之前,根据预设的 schema 对参数进行验证。如果验证失败,直接抛出错误,避免无效数据进入业务逻辑。这让你的业务方法能够专注于核心逻辑,而不用操心那些繁琐的验证细节。

这些例子只是冰山一角。装饰器还能用于自动绑定

this

(比如在 React 类组件中,避免在构造函数中手动

bind

),或者给类添加一些元数据(比如路由信息、DI配置等)。它的核心价值在于,把那些“与业务逻辑无关但又必须存在”的功能,以一种非侵入、可复用的方式,优雅地附加到你的代码上。

实现一个自定义装饰器,需要注意哪些细节和陷阱?

自己动手写装饰器,其实并不复杂,但有些细节如果不注意,可能会踩到一些坑。这就像搭积木,虽然基本规则简单,但要搭出稳固又好看的结构,还是得讲究技巧。

1. 理解不同类型装饰器的签名:这是最基础也最关键的一点。装饰器并不是一个万能函数,它根据你装饰的目标类型(类、方法、属性、访问器、参数)接收不同的参数。

类装饰器 (Class Decorator):

(target: Function)

target

就是类的构造函数。你可以修改它,或者返回一个新的构造函数来替换它。方法装饰器 (Method Decorator):

(target: Object, propertyKey: string | symbol, descriptor: PropertyDescriptor)

target

: 类的原型对象(对于静态方法则是类构造函数本身)。

propertyKey

: 方法的名字。

descriptor

: 方法的属性描述符,包含

value

(方法本身),

writable

,

enumerable

,

configurable

等。你可以修改

descriptor.value

来替换原方法,或者修改其他属性。别忘了返回修改后的

descriptor

属性装饰器 (Property Decorator):

(target: Object, propertyKey: string | symbol)

target

: 类的原型对象。

propertyKey

: 属性的名字。注意: 属性装饰器不能修改属性的

descriptor

,因为在装饰器执行时,属性还没有被初始化。它主要用于添加元数据。访问器装饰器 (Accessor Decorator):

(target: Object, propertyKey: string | symbol, descriptor: PropertyDescriptor)

。和方法装饰器类似,用于

getter

setter

参数装饰器 (Parameter Decorator):

(target: Object, propertyKey: string | symbol, parameterIndex: number)

target

: 类的原型对象。

propertyKey

: 方法的名字。

parameterIndex

: 参数在方法参数列表中的索引。主要用于添加元数据。

我个人觉得,最常用也最强大的还是方法装饰器,因为它能直接操作方法的行为。

2. 装饰器工厂 (Decorator Factory):如果你需要给装饰器传递参数,那么你就需要创建一个“装饰器工厂”。它是一个函数,接收你的参数,然后返回真正的装饰器函数。

// 这是一个接收参数的日志装饰器工厂function logWithLevel(level = 'INFO') {  return function (target, propertyKey, descriptor) {    const originalMethod = descriptor.value;    descriptor.value = function (...args) {      console.log(`[${level}] Calling method: ${propertyKey.toString()} with args: ${JSON.stringify(args)}`);      return originalMethod.apply(this, args);    };    return descriptor;  };}class Task {  @logWithLevel('DEBUG')  doSomething(data) {    console.log('Doing something with:', data);  }  @logWithLevel() // 默认INFO  finishTask() {    console.log('Task finished.');  }}const task = new Task();task.doSomething('payload');task.finishTask();

这里的

logWithLevel

就是装饰器工厂,它返回的匿名函数才是真正的装饰器。

3.

this

上下文的问题:当你用装饰器包装一个方法时,尤其是使用

descriptor.value = function(...)

这种方式时,一定要小心

this

的指向。在新的包装函数里,

this

可能会丢失原来的上下文。所以,通常你需要使用

originalMethod.apply(this, args)

或者

originalMethod.call(this, ...args)

来确保

originalMethod

在正确的

this

上下文中执行。这是一个非常常见的陷阱,我以前也在这里栽过跟头。

4. 装饰器的执行顺序:如果一个目标被多个装饰器装饰,它们的执行顺序是从下往上(对于同一行)或者从右往左(如果写在一行)。但最终的“包装”效果是自外向内。这意味着,最靠近目标定义的装饰器会最先执行,然后它的结果会被上一个装饰器接收并处理。理解这个顺序对于调试和预期行为至关重要。

5. 实验性特性:目前 JavaScript 的装饰器提案(TC39 Stage 3)仍在演进中,这意味着它的语法和行为在未来可能会有微小的变化。尽管 TypeScript 和 Babel 已经提供了支持,并且在实际项目中被广泛使用,但我们仍需意识到它不是一个最终定稿的 ECMAScript 标准。这通常意味着你需要一个构建工具(如 Babel)或 TypeScript 来转译你的代码。

6. 避免过度使用:装饰器虽好,但并非银弹。过度使用装饰器可能会让代码变得难以理解和调试,因为它隐藏了实际的逻辑流。有时候,简单的高阶函数或组合模式反而更清晰。选择合适的场景,让装饰器真正发挥其声明式、非侵入的优势,而不是为了用而用。

装饰器与高阶函数/高阶组件有什么异同,该如何选择?

这个问题问得好,因为在很多场景下,它们确实能解决类似的问题,都是关于“增强”现有功能。但它们在实现方式、适用场景和语义上,还是有挺大区别的。在我看来,它们就像是工具箱里不同形状的扳手,虽然都能拧螺丝,但有些螺丝用特定的扳手会更顺手。

高阶函数 (Higher-Order Functions, HOF):

定义: 接受一个或多个函数作为参数,或者返回一个函数作为结果的函数。这是函数式编程的核心概念。特点: 非常灵活,纯粹的函数组合。不依赖任何特殊语法。例子:

map

,

filter

,

reduce

都是 HOF。你自己写的

withLogger(func)

这种模式也是。优势: 极高的灵活性和可组合性,易于测试,函数式编程范式。劣势: 当需要层层包装时,代码可能会出现“回调地狱”或“包装地狱”,可读性会下降。

// HOF 示例function withLogger(fn) {  return function (...args) {    console.log(`[HOF Log] Calling ${fn.name} with args:`, args);    const result = fn.apply(this, args);    console.log(`[HOF Log] ${fn.name} returned:`, result);    return result;  };}function add(a, b) {  return a + b;}const loggedAdd = withLogger(add);loggedAdd(10, 20);

高阶组件 (Higher-Order Components, HOC):

定义: 在 React 生态系统中,HOC 是一个函数,它接受一个组件作为参数,并返回一个新组件。特点: 专门用于 React 组件的复用逻辑,例如数据获取、权限控制、状态管理等。例子:

withRouter

,

connect

(来自 Redux) 都是 HOC。优势: 强大的组件逻辑复用机制,能够注入 props 或修改组件行为。劣势: 同样可能导致“包装地狱”,使得组件树变得复杂;在 Hooks 出现后,很多 HOC 的场景被 Hooks 更好地替代了。

// HOC 示例 (React 伪代码)function withAuth(WrappedComponent) {  return class extends React.Component {    render() {      // 假设这里有认证逻辑      const isAuthenticated = true; // 模拟      if (!isAuthenticated) {        return 

请登录

; } return ; } };}class MyComponent extends React.Component { render() { return

欢迎, {this.props.currentUser.name}

; }}const AuthMyComponent = withAuth(MyComponent);//

装饰器 (Decorators):

定义: 一种特殊的语法糖,用于声明式地增强类、方法、属性等定义。特点: 语法简洁 (

@

符号),直接作用于定义,而非运行时包装。它本身就是一种特殊的 HOF,只不过是作用于类/方法层面。优势: 声明性强,代码意图清晰,减少样板代码,非常适合元编程和横切关注点的处理。劣势: 实验性特性(尽管广泛使用),特定于类/方法上下文,不适用于纯函数。

该如何选择?

我觉得,选择哪种方案,主要取决于你的目标、上下文以及你所处的生态系统。

如果你在处理类和类的方法: 装饰器往往是最佳选择。它以一种非常优雅、声明式的方式,直接在定义点附近增强功能。比如日志、性能监控、权限检查、自动绑定

this

等,用装饰器会比手动 HOF 包装清晰很多。它让你的代码看起来就像是“自描述”的。

如果你在进行纯函数式编程,或者需要高度灵活的函数组合: HOF 是你的不二之选。它们不依赖任何特殊语法,能够以非常细粒度的方式组合函数。例如,数据转换管道(

compose(f, g, h)

)就非常适合 HOF。

如果你在 React 组件中复用逻辑,并且你的项目还在使用类组件: HOC 仍然是一个有效的模式,尤其是在 Hooks 出现之前。但现在,React Hooks 已经能够解决 HOC 的大部分问题,并且以更简洁、更直接的方式。所以,在新的 React 项目中,HOC 的使用频率已经大大降低了。

混合使用: 很多时候,你可能需要混合使用这些模式。比如,你可能用装饰器来处理类方法级别的横切关注点,然后用 HOF 来处理一些纯函数的数据转换。在 React 中,你甚至可以用装饰器来简化 HOC 的应用(比如

@withAuth

装饰器直接应用到类组件上)。

总结一下,装饰器提供了一种结构化的、声明式的元编程能力,特别适合于在类和方法层面注入通用逻辑。而高阶函数则提供了更底层的、更灵活的函数组合能力,适用于任何函数。HOC 则是高阶函数在 React 组件层面的特定应用。没有绝对的好坏,只有最适合你当前场景的方案。关键在于理解它们的本质和适用边界,然后做出明智的选择。

以上就是JS 装饰器模式实战 – 使用 Decorators 增强类与方法的优雅方案的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1520523.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月20日 13:23:44
下一篇 2025年12月20日 13:23:56

相关推荐

  • TypeScript 中如何约束对象为 CSS 属性?

    typescript 中如何约束对象为 css 属性 想要约束一个对象为 css 属性,以便在调用函数时得到自动补全提示,可以采用以下方法: 使用 react 的 cssproperties 类型 对于 react 项目,可以使用 react 提供的 cssproperties 类型: 立即学习“前…

    2025年12月24日
    300
  • 如何在 TypeScript 中约束对象为 CSS 属性?

    如何在 typescript 中约束对象为 css 属性? 在 typescript 中,为特定目的而约束对象类型是很重要的。在本文中,我们将探究如何将对象约束为包含 css 属性。 考虑以下函数: function setattrstoelement(el: htmlelement, attr: …

    2025年12月24日
    000
  • 如何使用 TypeScript 约束对象以匹配 CSS 属性?

    如何约束 typescript 对象以匹配 css 属性? setattrstoelement 函数接收两个参数,其中第二个参数应为 css 属性。对于 react 项目,可以使用 cssproperties 类型: import { cssproperties } from “react”;fun…

    2025年12月24日
    000
  • 为什么使用 :global 修改 Antd 样式无效?

    :global 修改 antd 样式为何无效 本文旨在帮助您解决在组件内使用:global修改 antd 全局样式未生效的问题。 问题描述 您在组件内使用:global修改 antd 按钮样式,但没有生效。完整代码可参考 https://codesandbox.io/s/fk7jnl 。 解决方案 …

    2025年12月24日
    000
  • 为什么在 React 组件中无法获得 Tailwind CSS 语法提示?

    为什么在 React 组件中无法获得 Tailwind CSS 语法提示? 你在 VSCode 中编写 HTML 文件时,可以正常获取 Tailwind CSS 语法提示。但当你尝试在 React 组件中编写 Tailwind CSS 时,这些提示却消失不见了。这是什么原因造成的? 解决方案 要解决…

    2025年12月24日
    000
  • 如何在 VSCode 中为 React 组件启用 Tailwind CSS 提示?

    在 vscode 中为 react 组件启用 tailwind css 提示 如果你在使用 vscode 编写 react 组件时,发现 tailwind css 提示无法正常显示,这里有一个解决方法: 安装 tailwind css intellisense 插件 这是实现代码提示的关键,确保你已…

    2025年12月24日
    200
  • CSS 砌体 Catness

    css 就像技术中的其他东西一样 – 它总是在变化和发展。该领域正在进行的开发是 css 网格布局模块级别 3,也称为 css masonry 布局。 theo 制作了一段视频,介绍了它的开发方式以及苹果和谷歌就如何实施它进行的辩论。 所有这些让我很高兴尝试 css 砌体! webkit…

    好文分享 2025年12月24日
    000
  • 使用 React 构建 Fylo 云存储网站

    介绍 在这篇博文中,我们将逐步介绍如何使用 react 创建一个功能丰富的云存储网站。该网站受 fylo 启发,提供了主页、功能、工作原理、感言和页脚等部分。在此过程中,我们将讨论用于构建这个完全响应式网站的结构、组件和样式。 项目概况 该项目由多个部分组成,旨在展示云存储服务。每个部分都是用 re…

    2025年12月24日 好文分享
    000
  • 使用 React 构建食谱查找器网站

    介绍 在本博客中,我们将使用 react 构建一个食谱查找网站。该应用程序允许用户搜索他们最喜欢的食谱,查看趋势或新食谱,并保存他们最喜欢的食谱。我们将利用 edamam api 获取实时食谱数据并将其动态显示在网站上。 项目概况 食谱查找器允许用户: 按名称搜索食谱。查看趋势和新添加的食谱。查看各…

    2025年12月24日 好文分享
    200
  • 不可变数据结构:ECMA 4 中的记录和元组

    不可变数据结构:ecmascript 2024 中的新功能 ecmascript 2024 引入了几个令人兴奋的更新,但对我来说最突出的一个功能是引入了不可变数据结构。这些新结构——记录和元组——改变了 javascript 中数据管理的游戏规则。它们提供了一种令人满意的方式来保持我们的数据健全、安…

    2025年12月24日
    100
  • 为什么前端固定定位会发生移动问题?

    前端固定定位为什么会出现移动现象? 在进行前端开发时,我们经常会使用CSS中的position属性来控制元素的定位。其中,固定定位(position: fixed)是一种常用的定位方式,它可以让元素相对于浏览器窗口进行定位,保持在页面的固定位置不动。 然而,有时候我们会遇到一个问题:在使用固定定位时…

    2025年12月24日
    000
  • 从初学到专业:掌握这五种前端CSS框架

    CSS是网站设计中重要的一部分,它控制着网站的外观和布局。前端开发人员为了让页面更加美观和易于使用,通常使用CSS框架。这篇文章将带领您了解这五种前端CSS框架,从入门到精通。 Bootstrap Bootstrap是最受欢迎的CSS框架之一。它由Twitter公司开发,具有可定制的响应式网格系统、…

    2025年12月24日
    200
  • 克服害怕做选择的恐惧症:这五个前端CSS框架将为你解决问题

    选择恐惧症?这五个前端CSS框架能帮你解决问题 近年来,前端开发者已经进入了一个黄金时代。随着互联网的快速发展,人们对于网页设计和用户体验的要求也越来越高。然而,要想快速高效地构建出漂亮的网页并不容易,特别是对于那些可能对CSS编码感到畏惧的人来说。所幸的是,前端开发者们早已为我们准备好了一些CSS…

    2025年12月24日
    200
  • 深入理解CSS框架与JS之间的关系

    深入理解CSS框架与JS之间的关系 在现代web开发中,CSS框架和JavaScript (JS) 是两个常用的工具。CSS框架通过提供一系列样式和布局选项,可以帮助我们快速构建美观的网页。而JS则提供了一套功能强大的脚本语言,可以为网页添加交互和动态效果。本文将深入探讨CSS框架和JS之间的关系,…

    2025年12月24日
    000
  • 项目实践:如何结合CSS和JavaScript打造优秀网页的经验总结

    项目实践:如何结合CSS和JavaScript打造优秀网页的经验总结 随着互联网的快速发展,网页设计已经成为了各行各业都离不开的一项技能。优秀的网页设计可以给用户留下深刻的印象,提升用户体验,增加用户的黏性和转化率。而要做出优秀的网页设计,除了对美学的理解和创意的运用外,还需要掌握一些基本的技能,如…

    2025年12月24日
    200
  • is与where选择器:提升前端编程效率的秘密武器

    is与where选择器:提升前端编程效率的秘密武器 在前端开发中,选择器是一种非常重要的工具。它们用于选择文档中的元素,从而对其进行操作和样式设置。随着前端技术的不断发展,选择器也在不断演化。而其中,is与where选择器成为了提升前端编程效率的秘密武器。 is选择器是CSS Selectors L…

    2025年12月24日
    000
  • 前端技巧分享:使用CSS3 fit-content让元素水平居中

    前端技巧分享:使用CSS3 fit-content让元素水平居中 在前端开发中,我们常常会遇到需要将某个元素水平居中的情况。使用CSS3的fit-content属性可以很方便地实现这个效果。本文将介绍fit-content属性的使用方法,并提供代码示例。 fit-content属性是一个相对于元素父…

    2025年12月24日
    000
  • 前端技术分享:利用fit-content实现页面元素的水平对齐效果

    前端技术分享:利用fit-content实现页面元素的水平对齐效果 在前端开发中,实现页面元素的水平对齐是一个常见的需求。尤其在响应式布局中,我们经常需要让元素根据设备的屏幕大小自动调整位置,使页面更加美观和易读。在本文中,我将分享一种利用CSS属性fit-content来实现页面元素的水平对齐效果…

    2025年12月24日
    000
  • 学完HTML和CSS之后我应该做什么?

    网页开发是一段漫长的旅程,但是掌握了HTML和CSS技能意味着你已经赢得了一半的战斗。这两种语言对于学习网页开发技能来说非常重要和基础。现在不可或缺的是下一个问题,学完HTML和CSS之后我该做什么呢? 对这些问题的答案可以分为2-3个部分,你可以继续练习你的HTML和CSS编码,然后了解在学习完H…

    2025年12月24日
    000
  • 聊聊怎么利用CSS实现波浪进度条效果

    本篇文章给大家分享css 高阶技巧,介绍一下如何使用css实现波浪进度条效果,希望对大家有所帮助! 本文是 CSS Houdini 之 CSS Painting API 系列第三篇。 现代 CSS 之高阶图片渐隐消失术现代 CSS 高阶技巧,像 Canvas 一样自由绘图构建样式! 在上两篇中,我们…

    2025年12月24日 好文分享
    200

发表回复

登录后才能评论
关注微信