Object.defineProperty() 完整指南

Object.defineProperty() 完整指南

1. 基本概念

Object.defineProperty() 方法允许精确地添加或修改对象的属性。默认情况下,使用此方法添加的属性是不可修改的。

1.1 基本语法

javascript 复制代码
Object.defineProperty(obj, prop, descriptor)

参数说明:

  • obj: 要定义属性的对象
  • prop: 要定义或修改的属性名
  • descriptor: 属性描述符对象

2. 属性描述符

2.1 数据描述符

javascript 复制代码
const obj = {};

Object.defineProperty(obj, 'name', {
  value: 'John',          // 属性值
  writable: true,         // 是否可写
  enumerable: true,       // 是否可枚举
  configurable: true      // 是否可配置
});

2.2 访问器描述符

javascript 复制代码
const obj = {
  _name: 'John'
};

Object.defineProperty(obj, 'name', {
  get() {
    return this._name;
  },
  set(value) {
    this._name = value;
  },
  enumerable: true,
  configurable: true
});

3. 实际应用示例

3.1 数据劫持(Vue2响应式原理)

javascript 复制代码
function observe(obj) {
  if (typeof obj !== 'object' || obj === null) {
    return;
  }
  
  Object.keys(obj).forEach(key => {
    defineReactive(obj, key, obj[key]);
  });
}

function defineReactive(obj, key, val) {
  // 递归处理嵌套对象
  observe(val);
  
  Object.defineProperty(obj, key, {
    get() {
      console.log(`获取${key}属性`);
      return val;
    },
    set(newVal) {
      if (val === newVal) return;
      console.log(`设置${key}属性为${newVal}`);
      val = newVal;
      // 触发更新
    }
  });
}

// 使用示例
const data = {
  name: 'John',
  age: 20
};

observe(data);
data.name = 'Mike'; // 设置name属性为Mike
console.log(data.name); // 获取name属性 Mike

3.2 私有属性模拟

javascript 复制代码
function Person(name) {
  let _name = name;
  
  Object.defineProperty(this, 'name', {
    get() {
      return _name;
    },
    set(value) {
      if (typeof value !== 'string') {
        throw new Error('Name must be a string');
      }
      _name = value;
    }
  });
}

const person = new Person('John');
console.log(person.name); // John
person.name = 'Mike'; // 正常设置
person.name = 123; // 抛出错误

3.3 计算属性实现

javascript 复制代码
function computed(obj, key, computeFunc) {
  let value = computeFunc();
  
  Object.defineProperty(obj, key, {
    get() {
      return value;
    },
    set() {
      console.warn(`${key} is a computed property, cannot be modified`);
    }
  });
}

const obj = {
  a: 1,
  b: 2
};

computed(obj, 'sum', () => obj.a + obj.b);
console.log(obj.sum); // 3
obj.sum = 10; // 警告:sum is a computed property, cannot be modified

4. 注意事项和限制

4.1 不可扩展对象

javascript 复制代码
const obj = {};
Object.preventExtensions(obj);

// 这将抛出错误
Object.defineProperty(obj, 'name', {
  value: 'John'
});

4.2 继承属性

javascript 复制代码
const parent = {};
Object.defineProperty(parent, 'name', {
  value: 'John',
  writable: false
});

const child = Object.create(parent);
// 这将抛出错误
child.name = 'Mike';

4.3 属性描述符限制

javascript 复制代码
const obj = {};

// 不能同时指定 value/writable 和 get/set
Object.defineProperty(obj, 'name', {
  value: 'John',
  get() {
    return 'John';
  }
}); // 抛出错误

5. 性能考虑

5.1 大量属性处理

javascript 复制代码
// 不推荐
const obj = {};
for (let i = 0; i < 1000; i++) {
  Object.defineProperty(obj, `prop${i}`, {
    value: i,
    writable: true
  });
}

// 推荐
const descriptors = {};
for (let i = 0; i < 1000; i++) {
  descriptors[`prop${i}`] = {
    value: i,
    writable: true,
    configurable: true,
    enumerable: true
  };
}
Object.defineProperties(obj, descriptors);

5.2 访问器性能

javascript 复制代码
// 避免在访问器中进行复杂计算
Object.defineProperty(obj, 'name', {
  get() {
    // 不推荐
    return complexCalculation();
  }
});

// 推荐:缓存计算结果
let cachedValue;
Object.defineProperty(obj, 'name', {
  get() {
    if (cachedValue === undefined) {
      cachedValue = complexCalculation();
    }
    return cachedValue;
  }
});

6. 最佳实践

  1. 描述符默认值
javascript 复制代码
// 记住默认值都是 false
Object.defineProperty(obj, 'name', {
  value: 'John'
  // writable: false
  // enumerable: false
  // configurable: false
});
  1. 使用 TypeScript 类型
typescript 复制代码
interface PropertyDescriptor {
  configurable?: boolean;
  enumerable?: boolean;
  value?: any;
  writable?: boolean;
  get?(): any;
  set?(v: any): void;
}
  1. 错误处理
javascript 复制代码
function safeDefineProperty(obj, prop, descriptor) {
  try {
    Object.defineProperty(obj, prop, descriptor);
    return true;
  } catch (error) {
    console.error(`Failed to define property ${prop}:`, error);
    return false;
  }
}

7. 总结

Object.defineProperty() 的关键点:

  1. 使用场景

    • 数据劫持
    • 私有属性模拟
    • 计算属性实现
    • 属性访问控制
  2. 注意事项

    • 描述符类型限制
    • 性能考虑
    • 继承关系处理
    • 错误处理
  3. 最佳实践

    • 合理使用缓存
    • 避免复杂计算
    • 注意默认值
    • 做好错误处理

10. 深入理解 Object.defineProperty()

10.1 基础概念详解

Object.defineProperty() 是 JavaScript 中用于在对象上定义新属性或修改现有属性的方法。它允许精确控制属性的特性。

javascript 复制代码
// 基本语法
Object.defineProperty(obj, prop, descriptor)

// 参数说明
// obj: 要定义属性的对象
// prop: 要定义或修改的属性名
// descriptor: 属性描述符对象

10.2 属性描述符详解

属性描述符分为两种类型:数据描述符和访问器描述符。

  1. 数据描述符的完整选项:
javascript 复制代码
const obj = {};
Object.defineProperty(obj, 'name', {
  value: 'John',          // 属性值
  writable: true,         // 是否可写
  enumerable: true,       // 是否可枚举
  configurable: true      // 是否可配置
});
  1. 访问器描述符的完整选项:
javascript 复制代码
const obj = {
  _name: 'John'
};
Object.defineProperty(obj, 'name', {
  get() {
    console.log('Getting value');
    return this._name;
  },
  set(value) {
    console.log('Setting value to', value);
    this._name = value;
  },
  enumerable: true,
  configurable: true
});

10.3 常见使用场景

  1. 只读属性:
javascript 复制代码
const obj = {};
Object.defineProperty(obj, 'readonly', {
  value: 'I am read-only',
  writable: false,
  enumerable: true,
  configurable: false
});

obj.readonly = 'New value'; // 无效
console.log(obj.readonly);  // 'I am read-only'
  1. 不可枚举属性:
javascript 复制代码
const obj = {};
Object.defineProperty(obj, 'hidden', {
  value: 'You cannot see me',
  enumerable: false
});

console.log(Object.keys(obj)); // []
console.log(obj.hidden);       // 'You cannot see me'
  1. 计算属性:
javascript 复制代码
const person = {
  firstName: 'John',
  lastName: 'Doe'
};

Object.defineProperty(person, 'fullName', {
  get() {
    return `${this.firstName} ${this.lastName}`;
  },
  set(value) {
    [this.firstName, this.lastName] = value.split(' ');
  }
});

console.log(person.fullName);    // 'John Doe'
person.fullName = 'Jane Smith';
console.log(person.firstName);   // 'Jane'
console.log(person.lastName);    // 'Smith'
  1. Vue 双向绑定实现:
javascript 复制代码
function observe(obj) {
  if (!obj || typeof obj !== 'object') return;

  // 遍历对象的每个属性
  Object.keys(obj).forEach(key => {
    let value = obj[key];
    let dep = new Dep(); // 依赖收集器

    Object.defineProperty(obj, key, {
      get() {
        // 收集依赖
        if (Dep.target) {
          dep.addDep(Dep.target);
        }
        return value;
      },
      set(newValue) {
        if (value === newValue) return;
        value = newValue;
        // 通知所有依赖进行更新
        dep.notify();
      }
    });

    // 递归观察子属性
    if (typeof value === 'object') {
      observe(value);
    }
  });
}

// 使用示例
const data = {
  user: {
    name: 'John',
    age: 20
  }
};

observe(data);
// 现在 data 对象的所有属性都是响应式的

10.4 注意事项和最佳实践

  1. 描述符限制:
javascript 复制代码
// 不能同时使用数据描述符和访问器描述符
Object.defineProperty(obj, 'prop', {
  value: 123,
  get() { return 123; } // 错误!
});
  1. 性能优化:
javascript 复制代码
// 批量定义属性
Object.defineProperties(obj, {
  prop1: {
    value: 123,
    writable: true
  },
  prop2: {
    get() { return this.prop1 * 2; }
  }
});
  1. 默认值处理:
javascript 复制代码
// 所有描述符属性默认为 false
Object.defineProperty(obj, 'prop', {
  value: 123
  // writable: false
  // enumerable: false
  // configurable: false
});
相关推荐
里欧跑得慢2 小时前
Flutter 导航路由:构建流畅的应用导航体验
前端·css·flutter·web
@二进制2 小时前
vue3+vant4+ts出现页面空白?甚至在App.vue的<template></template>中随便输入都无法显示?
前端·vue.js·typescript
桂森滨2 小时前
Vue3+Pinia+Vite+TS 还原高性能外卖APP项目 4️⃣首页开发
前端·typescript·vue
hong1616882 小时前
TypeScript类型断言
linux·javascript·typescript
xyx-3v2 小时前
qt创建新工程
开发语言·c++·qt
我就是马云飞2 小时前
大专毕业两年,我如何进入大厂,并逆袭八年的技术与认知成长
前端·程序员·全栈
小陈工2 小时前
Python Web开发入门(十六):前后端分离架构设计——从“各自为政”到“高效协同”
开发语言·前端·数据库·人工智能·python
欣然~2 小时前
FachuanHybridSystem 项目 Windows 完整安装启动文档
前端
anyup2 小时前
uView Pro 的主题系统有多强大?3 分钟设计 uni-app 企业级 UI 主题
前端·vue.js·uni-app