掌握JavaScript对象属性修饰符:精确控制你的代码

JavaScript对象属性修饰符:控制属性的行为和可访问性

JavaScript是一门强大而灵活的编程语言,它提供了多种方法来定义和操作对象的属性。其中一个重要的概念是属性修饰符,它们允许我们控制属性的行为和可访问性。

基本属性修饰符

JavaScript中的属性修饰符主要包括writableenumerableconfigurable。我们可以使用Object.defineProperty方法来设置这些属性。

writable(可写性)

writable属性决定了属性是否可以被重新赋值。当设置为false时,属性不可被重新赋值,否则可以。

ini 复制代码
var obj = {
  b: 2,
};

Object.defineProperty(obj, 'a', {
  value: 10,
  writable: false, // 不可重写
});

obj.a = 'abc'; // 这里会抛出错误,因为属性a不可重写
console.log(obj.a); // 输出仍然是10

enumerable(可枚举性)

enumerable属性决定了属性是否会出现在for...in循环中或通过Object.keys方法获取。当设置为false时,属性不可被遍历,否则可以。

javascript 复制代码
var obj = {
  b: 2,
};

Object.defineProperty(obj, 'a', {
  value: 10,
  enumerable: false, // 不可遍历
});

for (var key in obj) {
  console.log(key); // 这里不会输出属性a
}

var keys = Object.keys(obj);
console.log(keys); // 这里不会包含属性a

configurable(可配置性)

configurable属性决定了属性是否可以被删除或属性描述符是否可以被修改。当设置为false时,属性不可被删除或修改属性描述符,否则可以。

javascript 复制代码
var obj = {
  b: 2,
};

Object.defineProperty(obj, 'a', {
  value: 10,
  configurable: false, // 不可修改描述符本身
});

// 下面的代码会抛出错误,因为属性a不可被删除
delete obj.a;

Object.defineProperty(obj, 'a', {
  writable: true, // 这里会抛出错误,因为属性a不可被修改
});

obj.a = 'abc'; // 这里会抛出错误,因为属性a不可重写

使用getter和setter

除了基本的属性修饰符,JavaScript还允许我们使用getter和setter来定义属性的访问和赋值行为。这对于在访问或修改属性时执行特定操作非常有用。

javascript 复制代码
var obj = {};

Object.defineProperty(obj, 'a', {
  get: function () {
    return 123; // 获取属性a时总是返回123
  }, 
  set: function (val) {
    throw new Error(
      `属性a不能被重新赋值,你尝试赋值为${val}`
    );
  },
});

console.log(obj.a); // 输出123

obj.a = 'abx'; // 这里会抛出错误,因为属性a不可重写

在上面的示例中,我们使用getter来获取属性a的值,并使用setter来阻止对属性a的重新赋值。这允许我们在属性访问和赋值时执行自定义逻辑。

在class类中的使用

在ES6类(Class)的构造函数中定义属性的getter和setter方法,可以直接在构造函数中使用Object.defineProperty

javascript 复制代码
class MyClass {
  constructor() {
    this._a = 123;

    Object.defineProperty(this, 'a', {
      get: function () {
        return this._a;
      },
      set: function (val) {
        throw new Error(`属性a不能被重新赋值,你尝试赋值为${val}`);
      },
    });
  }
}

const instance = new MyClass();
console.log(instance.a); // 输出123
instance.a = 456; // 这里会抛出错误,因为属性a不可重写

当使用ES6类(Class)来定义对象时,可以更简洁地定义属性的getter和setter方法,而不需要使用Object.defineProperty。ES6引入了一种更友好的语法,可以在类中使用getset关键字来创建属性的getter和setter。

javascript 复制代码
class MyClass {
  constructor() {
    this._a = 123;
  }

  get a() {
    return this._a; // 这是属性a的getter方法
  }

  set a(val) {
    throw new Error(
      `属性a不能被重新赋值,你尝试赋值为${val}`
    );
  }
}

结合遇到的面试题

  • js怎么观察对象是否被调用呢?
  • 获取对象属性在什么情况下会报错?
  • 手写一个ObjectFreeze
相关推荐
JieE2122 小时前
LeetCode 226. 翻转二叉树|JS 递归超详细拆解,二叉树入门经典题
javascript·算法
JieE2122 小时前
LeetCode 104. 二叉树的最大深度|递归思路超详细拆解
javascript·算法
爱勇宝3 小时前
鸿蒙生态的下半场:开发者不只要能开发,还要能赚钱
android·前端·程序员
IT_陈寒6 小时前
SpringBoot这个自动配置坑我跳了三次
前端·人工智能·后端
kyriewen6 小时前
我用 AI 一周写完了整个项目,上线第一天就崩了——这是我踩过最贵的 5 个坑
前端·javascript·ai编程
Larcher6 小时前
AI Loop:让AI像人一样自主完成任务的核心机制
javascript·人工智能·设计模式
默_笙6 小时前
🃏 JS 只有 8 种数据类型,但我花了 2 天才搞懂 null 和 undefined 的区别
javascript
牧艺6 小时前
从零到协同:构建类飞书在线文档系统的五个技术重难点
前端·人工智能
jump_jump7 小时前
流式 HTML:从 htmx 片段装配到浏览器原生增量渲染
javascript·性能优化·前端工程化
红尘散仙7 小时前
想写一个像样的终端 App?试试把 React 的开发体验搬进 Rust TUI
前端·rust