0

0

JavaScript装饰器怎样增强函数功能【教程】

夢幻星辰

夢幻星辰

发布时间:2026-01-21 18:57:13

|

175人浏览过

|

来源于php中文网

原创

装饰器是运行时函数劫持,非语法糖;它通过修改object.defineproperty行为重写类成员描述符,在类定义完成时执行,不支持普通函数声明,需编译支持且多装饰器自下而上覆盖。

javascript装饰器怎样增强函数功能【教程】

装饰器不是语法糖,是运行时函数劫持

JavaScript 装饰器(@ 语法)目前仍处于 Stage 3 提案阶段,**原生不支持**。你看到的 @log@debounce 等写法,必须配合 Babel(@babel/plugin-proposal-decorators)或 TypeScript 编译才能生效。直接在浏览器控制台敲 @memoize function foo() {} 会报 SyntaxError: Unexpected token '@'

本质是:装饰器是一个接收目标、属性名、描述符的函数,它返回一个新描述符(或修改原描述符),从而改变函数行为。不是“加一层壳”,而是重写 Object.defineProperty 的行为。

  • 类方法装饰器参数顺序固定:target, propertyKey, descriptor
  • 装饰器执行时机在类定义完成但实例未创建时,**不是每次调用函数时执行**
  • Babel 默认使用 legacy 模式(类似 TS 的旧行为),若要匹配提案最新语义,需显式配置 decoratorsBeforeExport: true

手写 @throttle 装饰器要注意 this 和参数透传

节流装饰器常被误写成只处理无参函数。实际中,被装饰的方法往往依赖 this 上下文和不定参数,漏掉这两点会导致 TypeError: Cannot read property 'xxx' of undefined 或参数丢失。

正确写法必须显式绑定 this 并展开参数:

立即学习Java免费学习笔记(深入)”;

function throttle(ms) {
  return function(target, prop, descriptor) {
    const fn = descriptor.value;
    let lastTime = 0;
    descriptor.value = function(...args) {
      const now = Date.now();
      if (now - lastTime >= ms) {
        fn.apply(this, args); // ← 关键:this 和 args 都要透传
        lastTime = now;
      }
    };
  };
}
  • 不能用箭头函数重写 descriptor.value,否则 this 指向丢失
  • 不要在装饰器内部直接调用 fn(),那会脱离原始调用上下文
  • 如果装饰的是 getter/setter,要用 descriptor.getdescriptor.set,而非 value

装饰器无法修饰普通函数声明,只能用于类成员或导出函数

你不能这样写:

Video Ocean
Video Ocean

人人皆导演,让视频创作变得轻松自如

下载
@log
function apiCall() { ... }

这在任何环境下都不合法。ECMAScript 装饰器规范**只允许用于类元素**(方法、访问器、字段)或模块级导出声明(如 export @memoize function fetchData() {})。想给独立函数加能力,有两个务实选择:

  • 改用高阶函数:const throttledFetch = throttle(300)(fetchData)
  • 把函数包进类里再装饰:class Api { @throttle(300) fetch() { ... } }
  • TypeScript 用户可启用 experimentalDecorators + emitDecoratorMetadata,但仅限编译期支持,运行时仍靠 polyfill

多个装饰器叠加时,执行顺序容易反直觉

写成这样:

@A
@B
@C
method() {}

实际执行顺序是 C → B → A(从下到上),但每个装饰器对 descriptor.value 的修改是**累积覆盖**的。比如:

@logvalue 改成带日志的包装函数;@throttle 又把它替换成节流版——最终生效的是最上面那个装饰器返回的描述符。

  • 装饰器之间无自动组合逻辑,不会自动形成“日志+节流”复合行为
  • 若需组合,应手动封装:@compose(log, throttle(200)),或让一个装饰器接受多个中间件函数
  • React 中的 connect + memo 类装饰器组合,本质是 HOC 嵌套,和原生装饰器机制无关

真正麻烦的从来不是语法怎么写,而是搞清「谁在什么时候、以什么上下文、修改了哪一段描述符」。装饰器不是魔法,它是明确的元编程操作——每一步都得对得上 Object.getOwnPropertyDescriptor 返回的东西。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
TypeScript工程化开发与Vite构建优化实践
TypeScript工程化开发与Vite构建优化实践

本专题面向前端开发者,深入讲解 TypeScript 类型系统与大型项目结构设计方法,并结合 Vite 构建工具优化前端工程化流程。内容包括模块化设计、类型声明管理、代码分割、热更新原理以及构建性能调优。通过完整项目示例,帮助开发者提升代码可维护性与开发效率。

47

2026.02.13

TypeScript全栈项目架构与接口规范设计
TypeScript全栈项目架构与接口规范设计

本专题面向全栈开发者,系统讲解基于 TypeScript 构建前后端统一技术栈的工程化实践。内容涵盖项目分层设计、接口协议规范、类型共享机制、错误码体系设计、接口自动化生成与文档维护方案。通过完整项目示例,帮助开发者构建结构清晰、类型安全、易维护的现代全栈应用架构。

191

2026.02.25

什么是中间件
什么是中间件

中间件是一种软件组件,充当不兼容组件之间的桥梁,提供额外服务,例如集成异构系统、提供常用服务、提高应用程序性能,以及简化应用程序开发。想了解更多中间件的相关内容,可以阅读本专题下面的文章。

182

2024.05.11

Golang 中间件开发与微服务架构
Golang 中间件开发与微服务架构

本专题系统讲解 Golang 在微服务架构中的中间件开发,包括日志处理、限流与熔断、认证与授权、服务监控、API 网关设计等常见中间件功能的实现。通过实战项目,帮助开发者理解如何使用 Go 编写高效、可扩展的中间件组件,并在微服务环境中进行灵活部署与管理。

226

2025.12.18

登录token无效
登录token无效

登录token无效解决方法:1、检查token的有效期限,如果token已经过期,需要重新获取一个新的token;2、检查token的签名,如果签名不正确,需要重新获取一个新的token;3、检查密钥的正确性,如果密钥不正确,需要重新获取一个新的token;4、使用HTTPS协议传输token,建议使用HTTPS协议进行传输 ;5、使用双因素认证,双因素认证可以提高账户的安全性。

6605

2023.09.14

登录token无效怎么办
登录token无效怎么办

登录token无效的解决办法有检查Token是否过期、检查Token是否正确、检查Token是否被篡改、检查Token是否与用户匹配、清除缓存或Cookie、检查网络连接和服务器状态、重新登录或请求新的Token、联系技术支持或开发人员等。本专题为大家提供token相关的文章、下载、课程内容,供大家免费下载体验。

842

2023.09.14

token怎么获取
token怎么获取

获取token值的方法:1、小程序调用“wx.login()”获取 临时登录凭证code,并回传到开发者服务器;2、开发者服务器以code换取,用户唯一标识openid和会话密钥“session_key”。想了解更详细的内容,可以阅读本专题下面的文章。

1092

2023.12.21

token什么意思
token什么意思

token是一种用于表示用户权限、记录交易信息、支付虚拟货币的数字货币。可以用来在特定的网络上进行交易,用来购买或出售特定的虚拟货币,也可以用来支付特定的服务费用。想了解更多token什么意思的相关内容可以访问本专题下面的文章。

2105

2024.03.01

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

3

2026.03.11

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 5.9万人学习

国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1万人学习

React核心原理新老生命周期精讲
React核心原理新老生命周期精讲

共12课时 | 1.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号