JavaScript中的Reflect是ES6引入的用于统一和规范化运行时对象操作的工具集,提供get、set、has等标准化方法,与Proxy协作实现可控、可组合的元编程。

JavaScript 中的反射(Reflection)不是指“照镜子”,而是指程序在运行时检查、发现和操作自身结构的能力。比如,动态获取对象有哪些属性、判断某个方法是否存在、甚至在运行时修改对象行为——这些都属于反射范畴。ES6 引入的 Reflect 对象,就是为统一和规范化这类操作而设计的工具集,它和 Proxy 配合紧密,也替代了不少过去散落在 Object 或全局函数中的操作。
Reflect 提供了对对象底层操作的标准化方法
过去很多操作是命令式的(如 name in obj、delete obj.x、obj[x] = y),而 Reflect 把它们封装成函数调用形式,返回明确的布尔值或结果,更利于拦截、调试和元编程:
-
Reflect.get(obj, key, receiver)—— 安全读取属性,支持自定义 getter 和 receiver(用于 this 绑定) -
Reflect.set(obj, key, value, receiver)—— 安全赋值,返回 true/false 表示是否成功(严格模式下失败会静默返回 false,而非抛错) -
Reflect.has(obj, key)—— 等价于key in obj,但可被 Proxy 拦截 -
Reflect.deleteProperty(obj, key)—— 类似delete obj.key,但返回布尔值 -
Reflect.getOwnPropertyDescriptor(obj, key)—— 获取属性描述符,比Object.getOwnPropertyDescriptor更一致(后者对非对象参数会强制转对象,Reflect版本直接报错) -
Reflect.defineProperty(obj, key, desc)—— 返回布尔值(成功 true),而Object.defineProperty成功时不返回,失败才抛错
Reflect 支持构造、原型与枚举等元操作
它把一些原本隐式或语法级的操作显式化,让 Proxy 处理逻辑更完整:
-
Reflect.construct(target, args, newTarget?)—— 模拟new Foo(...args),支持指定构造器和 new.target -
Reflect.getPrototypeOf(obj)和Reflect.setPrototypeOf(obj, proto)—— 替代Object.getPrototypeOf和Object.setPrototypeOf,语义更清晰,且setPrototypeOf返回布尔值 -
Reflect.isExtensible(obj)、Reflect.preventExtensions(obj)—— 判断/禁止扩展对象,比Object版本更符合反射语义 -
Reflect.ownKeys(obj)—— 返回所有自有属性键(包括 Symbol),是Object.getOwnPropertyNames(obj).concat(Object.getOwnPropertySymbols(obj))的简洁替代,也是 Proxy 中ownKeys捕获器的默认行为依据
Reflect 与 Proxy 协作构成完整的元编程基础
Reflect 方法名和参数几乎与 Proxy 的 trap(捕获器)一一对应。这种设计让代理逻辑可以“转发”到默认行为,避免重复实现:
立即学习“Java免费学习笔记(深入)”;
- 例如 Proxy 的
get捕获器中,常用return Reflect.get(target, prop, receiver)来执行原生读取逻辑 - 这样既保留默认行为,又便于在前后插入日志、权限校验或响应式追踪等逻辑
- 如果不使用
Reflect,就得手动模拟各种边界(如原型链查找、getter 调用、receiver 绑定),容易出错
基本上就这些。Reflect 不是让你天天手写反射代码的工具,而是让“控制对象行为”这件事变得更可控、可组合、可预测。它低调,但关键时候很稳。











