JavaScript中对象的属性类型
JavaScript中,对象的属性有两种:数据属性和访问器属性。
数据属性
特性:
数据属性包括一个数据值的位置。在这个位置可以读取和写入值。数据属性有4个特性。
[[configurable]]:可配置。表示能否删除属性、能否修改属性的特性、能否把属性变为访问器属性。[[enumerable]]:可枚举。表示能否通过for-in循环进行遍历。[[writable]]:可写。表示能否修改属性的值。[[value]]:表示这个属性的数据值。
设置:
ES5提供了Object.defineProperty()方法,来设置属性的默认特性。该方法接收三个参数:属性所在对象、属性名字、属性描述符对象。其中,描述符对象用来设置configurable、enumerable、writable、value中的一个或多个值。
ES5也提供了Object.defineProperties方法,可以定义多个属性。
- 在设置了属性的特性后,任何尝试超过configurable或writable权限的操作,在非严格模式下将会被忽略,在严格模式下将会抛出错误。
- 一旦把属性定义为
configurable: false,就不能再变回configurable: true了。否则,无论是否处于严格模式,都会抛出TypeError错误。 - 例外:即便属性是
foncigurable: false,仍然可以把writable的状态由true改为false,但是无法由false改为true。
var person = {};
Object.defineProperty(person, 'name', {
configurable: false,
writable: false,
value: 'Nicholas'
});
// 测试:删除属性
delete person.name; // 由于configurable: false,删除失败
console.log(person.name); // Nicholas
// 测试:修改特性
Object.defineProperty(person, 'name', {
configurable: true
}); // 抛出错误:TypeError: Cannot redefine property: name
// 测试:修改属性的值
person.name = 'Greg'; // 由于writable: false,修改失败
console.log(person.name); // Nicholas
- 结合
configurable: false和writable: false,就可以创建一个真正的常量属性。 Object.seal()方法,会在一个现有对象上调用Object.preventExtensions(),并把所有属性标记为configurable: false。Object.freeze()方法,会在一个现有对象上调用Object.seal(),并把所有属性标记为writable: false。
读取:
ES5提供了Object.getOwnPropertyDescriptor()方法,来取得给定属性的描述符。该方法接收两个参数:属性所在对象、属性名字。返回值是一个对象,其属性有configurable、enumerable、writable、value。
还有一种区分枚举属性enumerable的方法是propertyIsEnumerable()。该方法不检查原型链。
- 在调用Object.defineProperty()方法时,如果不指定,configurable、enumerable、writable的默认值为false。
- 对于直接定义在对象上的属性,configurable、enumerable、writable的默认值为true,value的值为undefined。
// 使用Object.definedProperty设置的属性
var descriptor = Object.getOwnPropertyDescriptor(person, 'name');
console.log(descriptor.configurable); // false
console.log(descriptor.enumerable); // false
console.log(descriptor.writable); // false
console.log(descriptor.value); // Nicholas
// 直接定义在对象上的属性
var person2 = { name: "Greg" };
var descriptor2 = Object.getOwnPropertyDescriptor(person2, 'name');
console.log(descriptor2.configurable); // true
console.log(descriptor2.enumerable); // true
console.log(descriptor2.writable); // true
console.log(descriptor2.value); // Greg
// propertyIsEnumerable()方法
console.log(person.propertyIsEnumerable('name')); // false
console.log(person2.propertyIsEnumerable('name')); // true
关于可枚举属性
Object.keys(obj)会返回一个数组,包含所有可枚举属性。不检查原型链。Object.getOwnPropertyNames(obj)会返回一个数组,包含所有属性,无论是否枚举。不检查原型链。prop in obj会返回一个布尔值,判断对象的属性是否存在,无论是否枚举。检查原型链。obj.hasOwnProperty(prop)会返回一个布尔值,判断对象的指定属性是否存在,无论是否枚举。不会检查原型链。- 不可枚举的属性,不会出现在
for-in循环的遍历中。
var myObject = {};
Object.defineProperty(myObject, 'a', {
enumerable: true,
value: 2
});
Object.defineProperty(myObject, 'b', {
enumerable: false,
value: 3
});
// propertyIsEnumerable
console.log(myObject.propertyIsEnumerable('a')); // true
console.log(myObject.propertyIsEnumerable('b')); // false
// Object.keys()
console.log(Object.keys(myObject)); // ["a"]
// Object.getOwnPropertyNames()
Object.getOwnPropertyNames(myObject); // ["a", "b"]
// in操作符
console.log('a' in myObject); // true
console.log('b' in myObject); // true
console.log('toString' in myObject); // true
// hasOwnProperty()
console.log(myObject.hasOwnProperty('a')); // true
console.log(myObject.hasOwnProperty('b')); // true
console.log(myObject.hasOwnProperty('toString')); // false
// 遍历
for (var k in myObject) {
console.log( k, myObject[k] );
} // a 2
访问器属性
特性:
访问器属性有4个特性。对于访问器属性,JavaScript会忽略它们的writable和value特性,取而代之的该关心get和set属性。
[[configurable]]:表示能否删除属性、能否修改属性的特性、能否把属性变为访问器属性。[[enumerable]]:表示能否通过for-in循环进行遍历。直接定义在对象上的属性,默认值为true。[[get]]:读取属性时调用的函数。直接定义在对象上的属性,默认值为undefined。[[set]]:写入属性时调用的函数。直接定义在对象上的属性,默认值为undefined。
设置:
同样使用Object.defineProperty()方法和Object.defineProperties方法设置访问其属性。
- 不一定非要同时指定getter和setter函数。
- 只指定getter函数意味着属性不能写。尝试写入属性,在非严格模式下将会被忽略,在严格模式下将会抛出错误。
- 只指定setter函数意味着属性不能读。尝试读取属性,在非严格模式下将会返回undefined,在严格模式下将会抛出错误。
var book = {
_year: 2004,
edition: 1
};
Object.defineProperty(book, 'year', {
get: function() {
return this._year;
},
set: function(newYear) {
if (newYear > 2004) {
this._year = newYear;
this.edition += (newYear - 2004);
}
}
});
// 测试
book.year = 2005; // 注意:defineProperty中定义的属性名为year,而非_year
console.log(book.edition); // 2
另一种设置getter和setter函数的方法如下:
var myObject = {
// 定义getter
get foo() {
return 'You get: ' + this._foo_;
},
// 定义setter
set foo(val) {
this._foo_ = val;
}
}
// 测试
myObject.foo = 2;
console.log(myObject.foo); // You get: 2
读取:
同样使用Object.getOwnPropertyDescriptor()方法取得给定属性的描述符。
- 在调用Object.defineProperty()方法时,如果不指定,configurable、enumerable的默认值为false,get、set的默认值为undefined。
- 对于直接定义在对象上的属性,configurable、enumerable的默认值为true。
// 使用Object.definedProperty设置的属性
var descriptor = Object.getOwnPropertyDescriptor(book, 'year');
console.log(descriptor.configurable); // false
console.log(descriptor.enumerable); // false
console.log(typeof descriptor.get); // function
console.log(typeof descriptor.set); // function
// 直接定义在对象上的属性
var descriptor = Object.getOwnPropertyDescriptor(myObject, 'foo');
console.log(descriptor.configurable); // true
console.log(descriptor.enumerable); // true
console.log(typeof descriptor.get); // function
console.log(typeof descriptor.set); // function
参考:《JavaScript高级程序设计》、《你不知道的JavaScript(上卷)》
JavaScript中对象的属性类型的更多相关文章
- JavaScript中对象的属性
在JavaScript中,属性决定了一个对象的状态,本文详细的研究了它们是如何工作的. 属性类型 JavaScript中有三种不同类型的属性:命名数据属性(named data properties) ...
- 记录,javascript中对象的属性名是字符串,却可以不用引号
问题描述:今日看书,里面介绍js的对象的属性名是包括空字符串在内的所以字符串 问题来了,我们平时定义的对象如下,是没有引号""or’'的 var someone = { f ...
- javascript中对象的属性的特性
1.ES5的属性特性包括下面六个: configurable: 表示能否通过delete来删除属性从而重新定义属性,能够修改属性的特性,默认为true enumberable: 表示是否能通过for- ...
- JavaScript中对象的属性:如何遍历属性
for/in 语句循环遍历对象的属性. js中获取key得到某对象中相对应的value的方法:obj.key js中根据动态key得到某对象中相对应的value的方法有二: 一.var key = & ...
- javascript中对象字面量的理解
javascript中对象字面量与数组字面量 第一部分 我们知道JavaScript中的数据类型有基本数据类型和引用类型,其中Object类型就是非常常用的类型.那么如果创建一个Object类型的实例 ...
- 关于JavaScript中对象的继承实现的学习总结
一.原型链 JavaScript 中原型链是实现继承的主要方法.其主要的思想是利用原型让一个引用类型继承另一个引用类型的属性和方法.实现原型链有一种基本模式,其代码如下. function Super ...
- javascript中对象的深度克隆
记录一个常见的面试题,javascript中对象的深度克隆,转载自:http://www.2cto.com/kf/201409/332955.html 今天就聊一下一个常见的笔试.面试题,js中对象的 ...
- Javascript中对象的Obeject.defineProperty()方法-------------(ES5/个人理解)
在讲到Obeject.defineProperty()方法之前先得说明一下ECMAScript中有两种属性:数据属性和访问器属性. 两种属性存在的意义:描述对象属性(key)的一些特性,因为这些属性是 ...
- C#保留2位小数几种场景总结 游标遍历所有数据库循环执行修改数据库的sql命令 原生js轮盘抽奖实例分析(幸运大转盘抽奖) javascript中的typeof和类型判断
C#保留2位小数几种场景总结 场景1: C#保留2位小数,.ToString("f2")确实可以,但是如果这个数字本来就小数点后面三位比如1.253,那么转化之后就会变成1.2 ...
随机推荐
- jQuery考试
No1: 分析:首先A答案是正确的jQuery中删除元素的方法有a,c,d所以排除B,另外c是清空,d虽然能删除但是不能删除元素所绑定的事件等等. No2: 分析:A是正确的通过get(index)的 ...
- 【DP】【P2340】奶牛会展
传送门 Description 奶牛想证明它们是聪明而风趣的.为此,贝西筹备了一个奶牛博览会,她已经对N 头奶牛进行了面试,确定了每头奶牛的智商和情商. 贝西有权选择让哪些奶牛参加展览.由于负的智商或 ...
- Consul 入门
1. 什么是Consul? Consul 有很多组件,对于整体来说,它是一个服务发现和服务配置的工具,它提供了一下特性: 服务发现 健康检查 KV存储 多数据中心 2.安装Consul 以下是在 Ce ...
- Codeforces Round #402 (Div. 2) A B C sort D二分 (水)
A. Pupils Redistribution time limit per test 1 second memory limit per test 256 megabytes input stan ...
- udhcpd源码分析4--获取client报文及发包动作
1:重要的结构体 获取的报文是UDP的payload部分,结构体struct dhcpMessage描述了dhcp报文的结构. /* packet.h */ struct dhcpMessage { ...
- C#学习之泛型准备
想要把泛型搞明白,最好先弄明白下面的代码实例 本实例是建立了两个类,然后在类中可以添加任意类型的值,并且可以利用foreach语句读出 //第一个节点类,放在一个文件中 using System; u ...
- NYOJ 737DP
石子合并(一) 时间限制:1000 ms | 内存限制:65535 KB 难度:3 描述 有N堆石子排成一排,每堆石子有一定的数量.现要将N堆石子并成为一堆.合并的过程只能每次将相邻的 ...
- 深入探析 Rational AppScan Standard Edition 多步骤操作
序言 IBM Rational AppScan Standard(下文简称 AppScan)作为面向 Web 应用安全黑盒检测的自动化工具,得到业界的广泛认可和应用.很多人使用 AppScan 时都采 ...
- Android 命名规范 (转)
刚接触android的时候,命名都是按照拼音来,所以有的时候想看懂命名的那个控件什么是什么用的,就要读一遍甚至好几遍才知道,这样的话,在代码的审查和修改过程中就会浪费不少不必要的时间.如果就是我一个人 ...
- 图论:最短路-Dijkstra
Dijkstra+堆优化具有稳定的时间复杂度,在一些数据范围要求比较严格(准确来说是图比较苛刻)的时候能够保证稳定的时间复杂度 但是Dijkstra不能够解决负边权的问题,所以在使用的时候一定要仔细读 ...