(八)控制对象属性-Object.defineProperty

前言

创建对象有许多方式,最常用的就是使用字面量的方式来创建,如下

js 复制代码
let obj = {
  name: "obj",
  age: 18,
};

但如果要精准控制对象的每个属性,比如增加一个属性height,这个属性是否可以删除、修改、遍历等,只靠对象字面量这种方式创建就没办法做到了,于是要引出今天的主题Object.defineProperty

Object.defineProperty 详讲

语法规则

Object.defineProperty(obj, property, propertyDescriptorMap)

  • obj: 要定义属性的对象。
  • property : 一个字符串或 Symbol,指定了要定义或修改的属性键。
  • propertyDescript : 要定义或修改的属性的描述符。
    返回值与参数一obj指向相同,所以一般不会有特定的变量来接收。

属性描述符

Object.defineProperty第三个参数为属性描述符,属性描述符分两种:数据属性描述符访问器属性描述符

但是不论数据属性描述符还是访问器属性描述符都会有两个通用的可选键,configerableenumerable,在此我先把这两个可选键统称为通用属性描述符

数据属性描述符

数据属性描述符主要来控制对象属性是否可写 ,其有两个键writablevalue

  • writable:控制对象属性是否可修改,默认值为false
  • value:对象属性的值,默认值为undefined
js 复制代码
let obj = {
  name: "obj",
  age: 18,
};
Object.defineProperty(obj, "height", {
  writable: false,
  value: 1.85,
});
obj.height = 2
console.log(obj.height); //1.85

上面的案例可以看出,如果writablefalse则不能对height属性进行修改,如果放在严格模式下修改height属性值则会报错。

访问器属性描述符

访问器属性描述符是由 getter/setter 函数对描述的属性,其有两个键getset,这两个健对应的值都为函数类型。

  • get:读取对象属性时的拦截器,只要读取属性则会调用该函数,函数返回值被用作该属性的值。
  • set:设置对象属性时的拦截器,只要给属性赋值则会调用该函数,并把值传递给该函数。
js 复制代码
function Obj() {
  let height = null;
  Object.defineProperty(this, "height", {
    get: function () {
      console.log("对象height属性被读取");
      return height;
    },
    set: function (value) {
      console.log("对象height属性被设置");
      height = value;
    },
  });
}
let newObj = new Obj();
newObj.height = 1.9;  //log:对象height属性被设置
console.log(newObj.height);
/*log:对象height属性被读取    1.9*/

我觉得上面案例应该不需要解释什么,如果换成下面的案例会发生什么?

js 复制代码
let obj = {
  name: "obj",
};

Object.defineProperty(obj, "height", {
  get: function () {
    return obj.height;
  },

  set: function (val) {
    obj.height = val;
  },
});

obj.height = 2;

首先对obj.height赋值会调用setter,而setter里又会对obj.height赋值,结果又会调用了setter....一直循环下去,变成了死循环。

^v^ 这是我依然会犯的错误

通用属性描述符

通用属性描述符包含两种configurableenumerable

  • configurable:表示属性是否可以删除 属性和是否可以再次配置属性描述符
  • enumerable:是否可以遍历该属性,如果为false,打印obj对象则不会显示该属性。使用Object.keys也不会显示该属性。使用Object.values也不会获取到对应的值。但该属性是真实存在的,只能通过对象.属性来获取该值。
js 复制代码
let obj = {
  name: "obj",
};

Object.defineProperty(obj, "height", {
  configurable: false,
  value: 2,
});

delete obj.height;

console.log(obj.height);

Object.defineProperty(obj, "height", {
  configurable: false,
  writable: true,
  value: 2,
});

obj.height = 3;  // 异常:Uncaught TypeError: can't redefine non-configurable property "height"

上面案例可以看出如果给属性的configurable设置为false,则该属性删除是没有效果的,如果在严格模式下则会报错,接下来我们又打算对height属性重新定义属性描述符,发现会报错,不能再次给height属性配置属性描述符。

js 复制代码
let obj = {
  name: "obj",
};

Object.defineProperty(obj, "height", {
  enumerable:false,
  value: 2,
});

console.log(Object.keys(obj));  //Array [ "name" ]

console.log(Object.values(obj)); //Array [ "obj" ]

console.log(obj.height); //2

该案例展示了height为不可遍历状态

属性描述符注意事项

数据属性描述符和访问器属性描述符不可以同时存在。也就是valuewritable不能与getset同时出现,但可以与通用属性描述符同时出现。

configurable enumerable writable value get set
数据属性描述符 可以 可以 可以 可以 不可以 不可以
访问器属性描述符 可以 可以 不可以 不可以 可以 可以

思考

js 复制代码
let obj = {
  name: "obj",
};

字面量定义的对象,obj.name都有什么属性描述符?

可以通过Object.getOwnPropertyDescriptor(obj, "name")输出看下。

相关推荐
非著名架构师1 小时前
js混淆的方式方法
开发语言·javascript·ecmascript
多多米10052 小时前
初学Vue(2)
前端·javascript·vue.js
敏编程2 小时前
网页前端开发之Javascript入门篇(5/9):函数
开发语言·javascript
看到请催我学习2 小时前
内存缓存和硬盘缓存
开发语言·前端·javascript·vue.js·缓存·ecmascript
XiaoYu20024 小时前
22.JS高级-ES6之Symbol类型与Set、Map数据结构
前端·javascript·代码规范
儒雅的烤地瓜4 小时前
JS | JS中判断数组的6种方法,你知道几个?
javascript·instanceof·判断数组·数组方法·isarray·isprototypeof
道爷我悟了4 小时前
Vue入门-指令学习-v-on
javascript·vue.js·学习
27669582924 小时前
京东e卡滑块 分析
java·javascript·python·node.js·go·滑块·京东
PleaSure乐事4 小时前
【Node.js】内置模块FileSystem的保姆级入门讲解
javascript·node.js·es6·filesystem
雷特IT5 小时前
Uncaught TypeError: 0 is not a function的解决方法
前端·javascript