JavaScript利用反射实现方法注入
1. 引言
反射是一种能够在程序运行时动态访问、修改某个类(对象)中属性和方法的机制
JavaScript在ES6中提供了Reflect 这一个内置的对象,它提供拦截 JavaScript 操作的方法
与之功能类似的有 proxy handler (en-US) 与Object.getOwnPropertyDescriptor()等
以下记述利用反射来实现对JavaScript对象的方法注入,主要使用Reflect和Object.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利用反射实现方法注入的更多相关文章
- 利用反射调用方法时,处理ref,out参数需要注意的问题(转)
转自:http://www.68idc.cn/help/buildlang/ask/20150318283817.html 项目中如下的泛型方法,因为要在运行时,动态指定类型参数,所以要利用反射来实现 ...
- laravel中如何利用反射实现依赖注入
依赖注入 在一个类中经常会依赖于其他的对象,先看一下经典的写法 class Foo { public $bar; public function __construct() { $this->b ...
- asp.net一般处理程序利用反射定位方法
asp.net的一般处理程序我想大家用得都不少,经常会如下如下的代码: using System; using System.Collections.Generic; using System.Lin ...
- javascript利用拷贝的方法实现导出excel(可以导出表格线)
Js代码: <script language=javascript> function preview() { window.clipboardData.setData("Tex ...
- java工具类-接受请求参数,并利用反射调用方法
public String a(HttpServletRequest request,HttpServletResponse response) throws JSONException, IOExc ...
- Python基础篇【第3篇】: Python异常处理、反射、动态导入、利用反射的web框架
异常处理 什么是异常? 异常即是一个事件,该事件会在程序执行过程中发生,影响了程序的正常执行. 一般情况下,在Python无法正常处理程序时就会发生一个异常.异常是Python对象,表示一个错误.当P ...
- 反射main方法
利用Java反射机制去调用其他类的main方法基于这种情形: 当程序中的某个类在运行到某处需要去调用其他类的main方法时,如果此程序并不知道此main方法所属类的名称,而只是在程序中接受某一代表此m ...
- C#利用反射调用PB编译的COM组件
问题: 1.根据COM组件的ProgID,得到COM组件公开的类型 2.创建COM组件提供的类型的对象 3.调用执行方法 正确姿势 C#利用反射调用(后期绑定)PB编译的COM组件 C#调用COM组件 ...
- <五>JDBC_利用反射及JDBC元数据编写通用的查询方法
此类针对javaBean类写了一个通用的查询方法,List<javaBean> 通用查询更新中...:通过学习,深刻体会到学会反射就等于掌握了java基础的半壁江山! 一.使用JDBC驱动 ...
- 利用反射及JDBC元数据编写通用查询方法
元数据:描述数据的数据,ResultSetMetaData是描述ResultSet的元数据对象,从它可以得到数据集有多少了,每一列的列名... ResultSetMetaData可以通过ResultS ...
随机推荐
- [Kafka]Kafka学习 -- 初识Kafka
Kafka学习 -- 初识Kafka 参考资料:稀土掘金<图解Kafka之实战指南>https://juejin.cn/book/6844733793220165639 Kafka是一个多 ...
- JS leetcode 反转字符串 题解分析
壹 ❀ 引 今天做的一道题非常简单,原题来自leetcode第344题反转字符串,题目如下: 编写一个函数,其作用是将输入的字符串反转过来.输入字符串以字符数组 char[] 的形式给出. 不要给另外 ...
- 使用JS访问本地数据库
1 前言 有时候,数据业务比较大,比如查询百万级的数据,如果使用JSP查询数据库,JSP的返回结果一般放在域名后面返回给客户端,而返回结果的长度是有限制的,数据过长可能会丢失部分数据:另一方面数据量大 ...
- js与java使用AES加密算法实现前后端加密解密
AES加密算法入门:https://blog.csdn.net/IndexMan/article/details/87284833 第三方crypto.js下载地址:https://download. ...
- 《深入理解Java虚拟机》(四) 调优工具、指令
目录 JVM 调优的概念 jps 1.options 功能选项 2.hostid jstat 1.vmid格式 2.interval 和 count 3.option jinfo jmap jhat ...
- vs 工程中替换 Qt 静态库
上篇介绍了如何编译 Qt 静态库 编译 windows 上的 qt 静态库 这篇介绍如何替换已有的 Qt 静态库,比如 Qt5.15.0 有很多 bug,我们不得不提升 Qt 版本来避免 bug 导致 ...
- docker自定义bridge网络
>>> docker network create -d bridge bridge-net # 创建一个名为bridge-net的网络 # 测试,启动两个容器,并且接入到bridg ...
- celery正常启动后能接收任务但不执行(已解决)
错误截图:celery接收到任务却不执行(多出在windows系统中) 解决方法1 添加–pool=solo参数 celery -A celery_tasks.main worker --pool=s ...
- 如何在矩池云复现开源对话语言模型 ChatGLM
ChatGLM-6B 是一个开源的.支持中英双语的对话语言模型,基于 General Language Model (GLM) 架构,具有 62 亿参数.结合模型量化技术,用户可以在消费级的显卡上进行 ...
- 51从零开始用Rust编写nginx,江湖救急,TLS证书快过期了
wmproxy wmproxy已用Rust实现http/https代理, socks5代理, 反向代理, 负载均衡, 静态文件服务器,websocket代理,四层TCP/UDP转发,内网穿透等,会将实 ...