」工欲善其事,必先利其器。「—孔子《論語.錄靈公》
首頁 > 程式設計 > JavaScript 裝飾器和自動存取器

JavaScript 裝飾器和自動存取器

發佈於2024-08-01
瀏覽:832

JavaScript Decorators and Auto-Accessors

如何创建 JavaScript 装饰器以及使用自动访问器如何帮助改善开发人员体验的演练。

目录

  • 上下文和规范
  • 前言
  • 自动访问器
  • 创建装饰器
    • 一个简单的装饰器
    • 使用装饰器进行验证
    • 装饰器选项
  • 元数据

背景和规范

GitHub 上的装饰器提案已经在分解装饰器的基本用例方面做得很好。我的目标不是在那里重新创建这些示例,而是突出一些鲜为人知的功能和交互。此外,在本系列的下一篇文章中,我将重点介绍如何在单个类属性上组合或链接多个装饰器。

前言

每个代码示例都将附带一个交互式 Babel REPL 游乐场的链接,因此您可以自己尝试,而无需设置 polyfill 或启动存储库。在我的所有示例中,应选中左上角(“设置”下)的“评估”选项,这意味着您将能够查看代码、编辑它、打开浏览器的开发控制台,并在那里查看日志/结果。

你不需要关注 Babel REPL 右侧的转译代码,除非你想深入研究装饰器的 polyfill。 Babel REPL 的左侧是您可以编辑和编写代码来亲自尝试的地方。

要强调的是,您的开发人员工具的控制台应该显示控制台日志。如果没有,请确保在左上角选中“评估”。

自动存取器

装饰器规范的一个重要功能是自动访问器。我们将从学习它们是什么以及使用自动访问器如何使编写装饰器变得更容易开始。

装饰器提案在此处概述了自动访问器。但最终它只是一个简单的功能;让我们看一个基本的工作示例:Babel REPL。

class MyClass {
  accessor myBoolean = false
}

在此类定义中,访问器关键字位于属性名称之前。然而,这并没有真正改变属性的任何内容 - 接下来,我们将看到自动访问器与装饰器结合使用时有多么有用。

(注意,您还可以将 static 与自动访问器一起使用,例如静态访问器 myBoolean = false)

创建装饰器

为了更好地理解为什么我们使用自动访问器,让我们构建一些装饰器。

一个简单的装饰器

我们首先将自动访问器与实际上作用不大的装饰器结合起来,以便了解语法。

这是一个保留内部变量的函数装饰器,并允许您通过类上的属性获取和设置该变量:Babel REPL

function simpleDecorator(value, context) {
  let internalValue = false
  return {
    get() {
      return internalValue
    },
    set(val) {
      internalValue = val
      return internalValue
    }
  }
}

class MyClass {
  @simpleDecorator
  accessor myBoolean
}

这个装饰器返回一个对象,有两个方法:get()和set()。这就是自动访问器的装饰器如何“装饰”或将属性的 setter 和 getter 包装在一个地方;我们不必创建 simpleGetterDecorator 和 simpleSetterDecorator。相反,我们使用自动访问器将它们组合成一个定义,这更容易。

最后,到目前为止,这看起来是一个相当正常的函数 - 这非常适合介绍!

使用装饰器进行验证

为了让我们为本文的其余部分做好准备,让我们更新我们的装饰器,以便它实际上执行某种验证。我们将制作一个装饰器,它只允许您设置偶数,而不允许设置其他数字。看起来像这样:Babel REPL

function onlyEvenNumbers(value, context) {
  let internalNumber = 0
  return {
    get() {
      return internalNumber
    },
    set(val) {
      const num = Number(val)
      if(isNaN(num)) {
        // don't set the value if it's not a number or coerced to a number
        return internalNumber
      }
      if(num % 2 !== 0) {
        // don't allow odd numbers
        return internalNumber
      }
      internalNumber = val
      return internalNumber
    }
  }
}

class MyClass {
  @onlyEvenNumbers
  accessor myEvenNumber
}

因此,我们向 set() 方法添加逻辑,现在任何尝试在我们的类上设置 myEvenNumber 属性的人都将经历该验证逻辑。好的。

装饰器选项

现在我们有了一个很好的仅偶数装饰器,让我们让它处理偶数和奇数,并提供一个选项来配置我们想要的数字类型!

幸运的是,因为我们在这里编写的是看起来相当正常的 JavaScript,所以将其配置为以这种方式工作并不难。我们用一个接受选项的函数包装原始装饰器,然后返回装饰器。巴别塔 REPL

function evensOrOdds(onlyEvens = true) {
  return function decorator(value, context) {
    let internalNumber = 0
    return {
      get() {
        return internalNumber
      },
      set(val) {
        const num = Number(val)
        if(isNaN(num)) {
            // don't set the value if it's not a number
            return internalNumber
        }
        if(num % 2 !== (onlyEvens ? 0 : 1)) {
            return internalNumber
        }
        internalNumber = val
        return internalNumber
      }
    }
  }
}

class MyClass {
  @evensOrOdds(true)
  accessor myEvenNumber

  @evensOrOdds(false)
  accessor myOddNumber
}

我们现在已经将装饰器配置为接受任意选项,这允许装饰器的用户自定义其行为。耶。

元数据

装饰器可以使用的另一个工具是 context.metadata。该对象被传递给每个装饰器,您可以将其用于各种用途,但您需要小心,因为元数据对象对于每个装饰器的所有调用都是相同的。

继续学习

继续阅读本系列的下一篇文章,了解如何将装饰器组合(或应用多个)到单个属性!

版本聲明 本文轉載於:https://dev.to/frehner/javascript-decorators-and-auto-accessors-437i?1如有侵犯,請聯絡[email protected]刪除
最新教學 更多>
  • 使用 React 建構的排序演算法視覺化工具
    使用 React 建構的排序演算法視覺化工具
    大家好!我剛剛完成了我的第一個真正的寵物專案 - 一個使用 React 構建的排序演算法視覺化工具。 ? GitHub 倉庫:https://github.com/Fedorse/Sorting-Algorithm-Visualizer 現場示範:https://algorithms-virid....
    程式設計 發佈於2024-11-06
  • 為 Angular 18 創建基本框架
    為 Angular 18 創建基本框架
    Ранее рассматривалось создание и настройка нового проекта Angular. В данной статье разберем базовую структуру. Напомню, что цикл посвящен разработке ...
    程式設計 發佈於2024-11-06
  • 如何存取Go的CGo中的聯合字段?
    如何存取Go的CGo中的聯合字段?
    在Golang CGo 中將Union 字段轉換為Go 類型在Golang CGo 中使用C 結構時,訪問union 字段可能是一個挑戰。常見場景涉及存取 C 結構內值聯合中的 ui32v 字段,如下例所示:struct _GNetSnmpVarBind { guint32 *oid...
    程式設計 發佈於2024-11-06
  • 在 JavaScript 中使用最小和最大堆管理流資料:數位運動員健康技術視角
    在 JavaScript 中使用最小和最大堆管理流資料:數位運動員健康技術視角
    数据管理在健康技术中至关重要。无论是跟踪运动员的表现指标还是监控运动员的恢复时间,有效地组织数据都可以对洞察的获取方式产生重大影响。在这种情况下管理数据的一种强大工具是堆,特别是最小堆和最大堆。在这篇文章中,我们将使用与运动员数据管理相关的实际示例,探讨如何在 JavaScript 中实现和使用最小...
    程式設計 發佈於2024-11-06
  • 使用 Matplotlib 繪圖時,為什麼效能會受到影響以及可以採取什麼措施?
    使用 Matplotlib 繪圖時,為什麼效能會受到影響以及可以採取什麼措施?
    Matplotlib 圖庫的效能注意事項在評估不同的 Python 圖庫時,使用 Matplotlib 時可能會遇到效能問題。本文探討了 Matplotlib 繪圖速度緩慢的原因,並提供了提高其速度的解決方案。 速度緩慢的原因Matplotlib 效能緩慢主要源自於兩個因素:頻繁重繪: 每次呼叫Fi...
    程式設計 發佈於2024-11-06
  • S - 單一職責原則(SRP)
    S - 單一職責原則(SRP)
    Single Responsibility Principle(SRP) The Single Responsibility Principle(SRP) is the first of the SOLID principles, which plays an important ...
    程式設計 發佈於2024-11-06
  • 如何修復 PHP 透過 SSH 連接 MySQL 時的 mysqli_connect() 參數問題?
    如何修復 PHP 透過 SSH 連接 MySQL 時的 mysqli_connect() 參數問題?
    在 PHP 中透過 SSH 連接到 MySQL 伺服器使用 PHP 函數透過 SSH 建立與遠端 Linux 電腦上託管的 MySQL 資料庫的連接可能具有挑戰性。使用提供的程式碼時,可能會出現錯誤「mysqli_connect()期望參數6為字串,給定資源」。 理解問題程式碼嘗試使用mysqli_...
    程式設計 發佈於2024-11-06
  • 微服務項目
    微服務項目
    ⚙️微服務專案的靈感來自@sqshq「Alexander Lukyanchikov」的piggymetrics,但這個實作使用了PostgreSQL和更簡單的業務邏輯,這個專案的主要目標是展示微服務架構的範例。 TechStack:PostgreSQL、Spring、Docker 我正在考慮可以添...
    程式設計 發佈於2024-11-06
  • 優化 AWS ECS 的 Java 堆設置
    優化 AWS ECS 的 Java 堆設置
    我們在 AWS Elastic Container Service(ECS) Fargate 上執行多個 Java 服務 (Corretto JDK21)。每個服務都有自己的容器,我們希望使用為每個進程支付的所有可能的資源。但這些步驟可以應用於 EC2 和其他雲端。 服務正在運行批次作業,延遲並不...
    程式設計 發佈於2024-11-06
  • PHP 初學者必備知識:釋放網站的全部潛力
    PHP 初學者必備知識:釋放網站的全部潛力
    PHP基礎:釋放網站潛能PHP是強大的伺服器端腳本語言,廣泛用於建立動態網站。對於初學者來說,掌握PHP基礎知識至關重要。本文將提供一個全面的指南,涵蓋PHP編程的基本要素,並透過實戰案例鞏固理解。 安裝並設定PHP要開始使用PHP,您需要安裝PHP解釋器和相關的軟體。遵循以下步驟:- 下载并安装P...
    程式設計 發佈於2024-11-06
  • 如何確定 PHP 標頭的正確圖片內容類型?
    如何確定 PHP 標頭的正確圖片內容類型?
    確定PHP 標頭的圖像內容類型確定PHP 標頭的圖像內容類型使用Header() 函數從Web 根目錄之外顯示圖像時,用戶可能會遇到困惑關於指定的內容類型:image/png。然而,儘管內容類型固定,但具有各種擴展名的圖像(例如, JPG、GIF)仍然可以成功顯示。 $filename = base...
    程式設計 發佈於2024-11-05
  • ByteBuddies:使用 Python 和 Tkinter 建立互動式動畫寵物
    ByteBuddies:使用 Python 和 Tkinter 建立互動式動畫寵物
    大家好! 我很高興向大家介紹 ByteBuddies,這是一個用 Python 和 Tkinter 創建的個人項目,展示了互動式動畫虛擬寵物。 ByteBuddies 將引人入勝的動畫與使用者交互相結合,提供了展示 GUI 程式設計強大功能的獨特體驗。該項目旨在透過提供互動式虛擬寵物來讓您的螢幕充...
    程式設計 發佈於2024-11-05
  • 如何解決“TypeError:\'str\'物件不支援專案分配”錯誤?
    如何解決“TypeError:\'str\'物件不支援專案分配”錯誤?
    'str'物件項目分配錯誤疑難排解'str'物件項目分配錯誤疑難排解嘗試在Python 中修改字串中的特定字元時,您可能會遇到錯誤「類型錯誤:「str」物件不支援專案分配。」發生這種情況是因為Python 中的字串是不可變的,這意味著它們無法就地更改。 >>...
    程式設計 發佈於2024-11-05
  • 如何緩解 GenAI 程式碼和 LLM 整合中的安全問題
    如何緩解 GenAI 程式碼和 LLM 整合中的安全問題
    GitHub Copilot and other AI coding tools have transformed how we write code and promise a leap in developer productivity. But they also introduce new ...
    程式設計 發佈於2024-11-05
  • Spring 中的 ContextLoaderListener:必要的邪惡還是不必要的複雜?
    Spring 中的 ContextLoaderListener:必要的邪惡還是不必要的複雜?
    ContextLoaderListener:必要的邪惡還是不必要的複雜? 開發人員經常遇到在 Spring Web 應用程式中使用 ContextLoaderListener 和 DispatcherServlet。然而,一個令人煩惱的問題出現了:為什麼不簡單地使用 DispatcherServle...
    程式設計 發佈於2024-11-05

免責聲明: 提供的所有資源部分來自互聯網,如果有侵犯您的版權或其他權益,請說明詳細緣由並提供版權或權益證明然後發到郵箱:[email protected] 我們會在第一時間內為您處理。

Copyright© 2022 湘ICP备2022001581号-3