1. 引言

反射是一种能够在程序运行时动态访问、修改某个类(对象)中属性和方法的机制

JavaScript在ES6中提供了Reflect 这一个内置的对象,它提供拦截 JavaScript 操作的方法

与之功能类似的有 proxy handler (en-US)Object.getOwnPropertyDescriptor()

以下记述利用反射来实现对JavaScript对象的方法注入,主要使用ReflectObject.getOwnPropertyDescriptor()来实现

2. Reflect

Reflect并非一个构造函数,所以不能通过 new 运算符对其进行调用,或者将 Reflect 对象作为一个函数来调,Reflect 的所有属性和方法都是静态的(就像 Math 对象),详细文档可参考MDN:Reflect - JavaScript | MDN (mozilla.org)

Reflect的主要功能就是对JavaScript对象进行操作,包括获取属性、修改属性等

在这里,要实现对JavaScript对象的原有方法进行注入,基本思路就是获取对象的原方法,设置一个新方法,在新方法中修改或调用原方法,最后将新方法替换原方法,示例代码如下:

// 定义一个对象
const obj = {
method: function () {
console.log('Original method');
}
}; // 获取对象的属性描述符
const descriptor = Reflect.getOwnPropertyDescriptor(obj, 'method'); // 保存原有方法的引用
const originalMethod = descriptor.value; // 修改方法的特性
descriptor.value = function () {
// 调用原有方法
Reflect.apply(originalMethod, this, arguments); console.log('Modified method');
}; // 定义修改后的方法
Reflect.defineProperty(obj, 'method', descriptor); // 调用方法
obj.method(); // 输出: Original method Modified method

3. Object

Object是JavaScript标准内置对象,其内置多个操作对象方法和属性的静态方法,包括获取属性、修改属性等,更为详细的文档可参考MDN:Object - JavaScript | MDN (mozilla.org)

在这里,要实现对JavaScript对象的原有方法进行注入,基本思路就是和上面是相同的:获取对象的原方法,设置一个新方法,在新方法中修改或调用原方法,最后将新方法替换原方法,示例代码如下:

// 定义一个对象
const obj = {
method: function () {
console.log('Original method');
}
}; // 获取对象的属性描述符
const descriptor = Object.getOwnPropertyDescriptor(obj, 'method'); // 保存原有方法的引用
const originalMethod = descriptor.value; // 修改方法的特性
descriptor.value = function () {
// 调用原有方法
originalMethod.apply(this, arguments); console.log('Modified method');
}; // 定义修改后的方法
Object.defineProperty(obj, 'method', descriptor); // 调用方法
obj.method(); // 输出: Original method Modified method

其实使用Object的静态方法与使用Reflect对象的静态方法差不多,没有太大区别

4. 示例

4.1 添加新方法

JavaScript内置的Date对象没有输出指定格式的方法,可以使用反射进行注入,实现对象的扩展,示例如下:

const getFormattedDate = function (date) {
const year = date.getFullYear();
const month = date.getMonth() + 1;
const day = date.getDate();
return `${year}-${month}-${day}`;
}; const date = new Date(); Reflect.defineProperty(date, 'getFormattedDate', {
value: getFormattedDate
}); console.log(date.getFormattedDate(date)); // 输出: '2023-7-24'

4.2 扩展原方法

JavaScript内置的Date.prototype.getMonth()方法,返回一个指定的日期对象的月份,为基于 0 的值(0 表示一年中的第一月)

const date = new Date();
console.log(date.getMonth()); // 输出: 6

现在修改为基于 1 的值(1 表示一年中的第一月)

    const date = new Date();

    const originalMethod = Reflect.get(date, 'getMonth');

    Reflect.defineProperty(date, 'getMonth', {
value: function () {
return originalMethod.apply(this, arguments) + 1;
}
}); console.log(date.getMonth()); // 输出: 7

5. 参考资料

[1] Reflect - JavaScript | MDN (mozilla.org)

[2] Object - JavaScript | MDN (mozilla.org)

[3] Date - JavaScript | MDN (mozilla.org)

JavaScript利用反射实现方法注入的更多相关文章

  1. 利用反射调用方法时,处理ref,out参数需要注意的问题(转)

    转自:http://www.68idc.cn/help/buildlang/ask/20150318283817.html 项目中如下的泛型方法,因为要在运行时,动态指定类型参数,所以要利用反射来实现 ...

  2. laravel中如何利用反射实现依赖注入

    依赖注入 在一个类中经常会依赖于其他的对象,先看一下经典的写法 class Foo { public $bar; public function __construct() { $this->b ...

  3. asp.net一般处理程序利用反射定位方法

    asp.net的一般处理程序我想大家用得都不少,经常会如下如下的代码: using System; using System.Collections.Generic; using System.Lin ...

  4. javascript利用拷贝的方法实现导出excel(可以导出表格线)

    Js代码: <script language=javascript> function preview() { window.clipboardData.setData("Tex ...

  5. java工具类-接受请求参数,并利用反射调用方法

    public String a(HttpServletRequest request,HttpServletResponse response) throws JSONException, IOExc ...

  6. Python基础篇【第3篇】: Python异常处理、反射、动态导入、利用反射的web框架

    异常处理 什么是异常? 异常即是一个事件,该事件会在程序执行过程中发生,影响了程序的正常执行. 一般情况下,在Python无法正常处理程序时就会发生一个异常.异常是Python对象,表示一个错误.当P ...

  7. 反射main方法

    利用Java反射机制去调用其他类的main方法基于这种情形: 当程序中的某个类在运行到某处需要去调用其他类的main方法时,如果此程序并不知道此main方法所属类的名称,而只是在程序中接受某一代表此m ...

  8. C#利用反射调用PB编译的COM组件

    问题: 1.根据COM组件的ProgID,得到COM组件公开的类型 2.创建COM组件提供的类型的对象 3.调用执行方法 正确姿势 C#利用反射调用(后期绑定)PB编译的COM组件 C#调用COM组件 ...

  9. <五>JDBC_利用反射及JDBC元数据编写通用的查询方法

    此类针对javaBean类写了一个通用的查询方法,List<javaBean> 通用查询更新中...:通过学习,深刻体会到学会反射就等于掌握了java基础的半壁江山! 一.使用JDBC驱动 ...

  10. 利用反射及JDBC元数据编写通用查询方法

    元数据:描述数据的数据,ResultSetMetaData是描述ResultSet的元数据对象,从它可以得到数据集有多少了,每一列的列名... ResultSetMetaData可以通过ResultS ...

随机推荐

  1. [Kafka]Kafka学习 -- 初识Kafka

    Kafka学习 -- 初识Kafka 参考资料:稀土掘金<图解Kafka之实战指南>https://juejin.cn/book/6844733793220165639 Kafka是一个多 ...

  2. JS leetcode 反转字符串 题解分析

    壹 ❀ 引 今天做的一道题非常简单,原题来自leetcode第344题反转字符串,题目如下: 编写一个函数,其作用是将输入的字符串反转过来.输入字符串以字符数组 char[] 的形式给出. 不要给另外 ...

  3. 使用JS访问本地数据库

    1 前言 有时候,数据业务比较大,比如查询百万级的数据,如果使用JSP查询数据库,JSP的返回结果一般放在域名后面返回给客户端,而返回结果的长度是有限制的,数据过长可能会丢失部分数据:另一方面数据量大 ...

  4. js与java使用AES加密算法实现前后端加密解密

    AES加密算法入门:https://blog.csdn.net/IndexMan/article/details/87284833 第三方crypto.js下载地址:https://download. ...

  5. 《深入理解Java虚拟机》(四) 调优工具、指令

    目录 JVM 调优的概念 jps 1.options 功能选项 2.hostid jstat 1.vmid格式 2.interval 和 count 3.option jinfo jmap jhat ...

  6. vs 工程中替换 Qt 静态库

    上篇介绍了如何编译 Qt 静态库 编译 windows 上的 qt 静态库 这篇介绍如何替换已有的 Qt 静态库,比如 Qt5.15.0 有很多 bug,我们不得不提升 Qt 版本来避免 bug 导致 ...

  7. docker自定义bridge网络

    >>> docker network create -d bridge bridge-net # 创建一个名为bridge-net的网络 # 测试,启动两个容器,并且接入到bridg ...

  8. celery正常启动后能接收任务但不执行(已解决)

    错误截图:celery接收到任务却不执行(多出在windows系统中) 解决方法1 添加–pool=solo参数 celery -A celery_tasks.main worker --pool=s ...

  9. 如何在矩池云复现开源对话语言模型 ChatGLM

    ChatGLM-6B 是一个开源的.支持中英双语的对话语言模型,基于 General Language Model (GLM) 架构,具有 62 亿参数.结合模型量化技术,用户可以在消费级的显卡上进行 ...

  10. 51从零开始用Rust编写nginx,江湖救急,TLS证书快过期了

    wmproxy wmproxy已用Rust实现http/https代理, socks5代理, 反向代理, 负载均衡, 静态文件服务器,websocket代理,四层TCP/UDP转发,内网穿透等,会将实 ...