JavaScript需要Proxy,是因为它提供了唯一原生、细粒度、可编程的方式拦截和自定义对象基本操作;而Object.defineProperty仅能静态劫持已存在属性,无法捕获新增/删除属性、in操作、数组索引访问等。
JavaScript 需要 Proxy,是因为它提供了**唯一原生、细粒度、可编程的方式去拦截和自定义对象的基本操作**(比如读取属性、赋值、枚举、函数调用等)。没有 Proxy,你无法真正“监听”或“改写”这些底层行为——Object.defineProperty 只能劫持已存在的属性,且无法捕获新增/删除属性、in 操作、for...in 遍历等;而 defineProperty 也无法代理数组索引访问或内置方法调用。
Proxy 通过一个“陷阱”(trap)机制,在目标对象被访问时插入自定义逻辑。常用且实用的拦截包括:
prop in obj,可隐藏属性或模拟稀疏结构Object.keys()、for...in、Reflect.ownKeys(),控制属性可见性new 操作,实现类代理或构造逻辑增强Proxy 不复制目标对象,也不修改原对象。它返回一个新对象(代理对象),所有操作都先经过你定义的 trap 函数,再转发给目标(或完全替代)。你可以选择放行、改写、拒绝,甚至返回完全不同类型的值。
例如:
const target = { x: 1 };
const proxy = new Proxy(target, {
get(obj, prop) {
console.log(`读取 ${prop}`);
return prop === 'y' ? 42 : obj[prop]; // 动态返回 y=42,不依赖原对象
}
});
console.log(proxy.x); // 输出 "读取 x" → 1
console.log(proxy.y); // 输出 "读取 y" → 42
Object.defineProperty 是“静态属性级”的控制,只能对已知属性名提前设置 get/set;而 Proxy 是“动态操作级”的拦截,不管属性是否存在、是否后续新增、是否是 Symbol、是否是数组索引,只要操作发生,就能捕获。
Proxy,之后所有属性读写都能拦截;用 defineProperty 则必须一个个定义push、length 变更、索引赋值(arr[0] = ...)等,Proxy 天然支持;defineProperty 对数组索引无效(除非手动为每个索引定义)Proxy 可拦截 deleteProperty、isExtensible、preventExtensions 等元操作,这是 defineProperty 完全做不到的除了 Vue/React 状态库这类典型场景,Proxy 还常用于:
eval、document 等危险对象的访问
set 中抛错,或返回新对象(配合 freeze 使用)