博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
JavaScript之面向对象的概念,对象属性和对象属性的特性简介
阅读量:5817 次
发布时间:2019-06-18

本文共 7599 字,大约阅读时间需要 25 分钟。

  一、大家都知道,面向对象语言有一个标志,那就是他们都有类的概念,通过类我们可以创建任意多个具有相同属性和方法的对象。但ECMAScript(指定JavaScript标准的机构,也就是说JavaScript是实现其标准的扩展)并没有类的概念,因此他的对象和基于类的语言中的对象有所不同,ECMAScript把对象定义为:"无需属性的集合,其属性可以包含基本值、对象或者函数"。严格的来说,这就相当于说对象是一组没有特定顺序的值。对象的每个属性或方法都有一个名字,而每个名字都映射到一个值。正应为这样,我们可以把ECMAScript的对象想象成散列表;无非就是一组键值对,其中值可以是数据或函数。每个对象都是基于一个引用类型创建的。

//JavaScript早期的对象定义方式        var person = new Object();        person.name = "张三";        person.age = 22;        person.job = "coder";        person.sayName = function () {            alert(this.name);        }        //几年后,对象字面量成为创建这种对象的首选模式        var person = {            name: "Nicholas",            age: 22,            job: "coder",            sayName: function () {                alert(this.name);            }        }        person.sayName();

注意:这里的对象,不像真正的面向对象语言那样,对象真的被创建了,这里对象的概念只是一个数据集合(这个数据集合可以存放任何数据)的引用,这个引用值不会改变,而面向对象的类你每new一次,他的引用值都会改变一次。

二、面向对象的属性类型

在ES5中在定义只有内部采用的特性时,描述了属性(property)的各种特征。ECMA-262定义这些特性是为了实现JavaScript引擎用的,因此在JavaScript中不能直接访问他们,为了表示特性是内部值,ECMA-262规范把它们放在了两对方括号中,例如[[Enumerable]]

ECMAScript中有两种属性:数据属性和访问器属性。

1、数据属性

数据属性包含一个数据值的位置。在这个位置可以读取和写入值。数据属性有4个描述其行为的特性。

(1)[[Configurable]]:表示能否通过delete删除属性从而重新定义属性,能否修改属性的特性,或者能否把属性修改为访问器属性。像上面代码中直接在对象上定义的属性,他们的这个特性默认值为true.

(2)[[Enumerable]]:表示能否通过for-in循环返回属性。像上面代码中直接在对象上定义的属性,他们的这个特性值默认为true。

(3)[[Writable]]:表示能否修改属性的值。像上面代码中直接在对象定义的属性,他们的这个特性默认为true。

(4)[[Value]]:包含这个属性的数据值。读取属性值的时候,从这个位置读;写入属性值的时候.把新值保存在这个位置。这个特性的默认值为undefined。也就是说如果你不给属性的该特性赋值,他的值将会是undefined。

现在有如下代码:

 

var person={     name:"张三"};

 

      像上面中直接在对象中定义的属性,他们的[[Configurable]]、[[Enumerable]]、[[Writable]]特性都被设置为true,而[[Value]]被设置为指定的值"张三";按照上面特性的描述,person对象中的name属性可以通过delete删除重新定义该属性,可以修改该属性的特性,可以把该属性修改为访问器属性。可以通过for-in循环返回属性,可以修改属性的值。

应为ECMA-262规范中提到属性的特性是为了实现JavaScript引擎所用到,所以我们不能通过JavaScript直接访问,但是JavaScript给我们提供了了一个方法,来操作我们需要操作的对象的属性的特性;这个方法是

 

//这个方法接收三个参数:属性所在的对象引用、属性的名字和一个描述符对象        //其中描述符对象的属性必须是上面提到的四个属性的特性(实现JavaScript引擎所用)        //Configurable、Enumerable、Writable、value        Object.defineProperty();

 

下面是这个方法的应用:

var person = {};        Object.defineProperty(person, "name", {          writable:false, //writable这个特性决定了当前属性的属性值能否被修改,这边设置为false,也就是不能被修改          value:"张三"      });      alert(person.name);      person.name = "李四";//所以给name重新赋值并没有效果,如果将writable的值修改为true,这边的赋值就会成功!      alert(person.name);

两次输出都是"张三"

 

alert(person.name);

 

var person = {};        Object.defineProperty(person, "name", {            configurable: true, //configurable这个特性决定了当前属性能否通过delete删除从而重新定义属性,能否修改属性的特性,能否将属性修改为访问器属性。          //这边设置为true,所以都可以          value:"张三"      });      alert(person.name);      delete person.name;//所以当这边删除name属性后,person对象就不存在了name属性      alert(person.name);//所以这边输出undefined因为此时person对象没有了name属性

 

输出:"张三","undefined";

 

 

var person = {};        Object.defineProperty(person, "name", {            configurable: false, //configurable这个特性决定了当前属性能否通过delete删除从而重新定义属性,能否修改属性的特性,能否将属性修改为访问器属性。          //这边设置为false,所以都不可以            value:"张三"      });      alert(person.name);      delete person.name;//所以当这边执行删除name属性的动作没有效果      alert(person.name);//所以这边输出还是"张三"

输出:"张三","张三"

var person = {};        Object.defineProperty(person, "name", {            configurable: false, //configurable这个特性决定了当前属性能否通过delete删除从而重新定义属性,能否修改属性的特性,能否将属性修改为访问器属性。          //这边设置为false,所以name属性被设置成为无法配置的属性          value:"张三",      });        Object.defineProperty(person, "name", {            configurable: true, //报错  value:"张三",      });
 

这边需要注意:当我们把属性定义为不给配置的之后,就不能再把它变回可配置的了。此时如果再调用Object.defineProperty()方法修改出writable之外的特性,都会导致错误。

输出:TypeError: can't redefine non-configurable property "name"

注意:当我们调用Object.defineProperty()方法,在指定了对象和对象的属性却没有指定描述符对象的的configurable、writable、Enumerbale的三个特性时,那么他们的默认值都为false;

综上所述:我们可以通过Object.defineProperty()方法来多次修改同一个属性,但是当我们把属性的configurable特性的值设置成false,就会有所限制了,我们只能修改属性的writable特性的值了;

 

2、访问器属性

访问器属性和数据属性的区别是:访问器属性不包含数据值,且类似与面向对象里面的类属性,他们都包含一对getter和setter函数,在读取访问器属性时,会调用getter函数,这个函数会返回有效的值,在写入访问器属性时,会调用setter函数并写入新值,这个函数负责决定如何处理数据,这个面向对象中的类属性大致一样!

访问器属性和数据属性一样,有4个特性:

[[Configurable]]:表示能否通过delete删除属性从而重新定义属性,能否修改属性的特性,或者能否把属性修改为数据属性,对于直接在对象上定义的属性,这个特性的默认值为true;

[[Enumerable]]:表示能否通过for-in循环返回属性。对于直接在对象上定义的属性。这个特性的默认值为true。

[[Get]]:在读取属性时调用的函数,默认值为undefined。

[[Set]]:在写入属性时调用的函数。默认值为undefined。

访问器属性不能直接定义,必须使用Object.definneProperty().请看下面代码:

var book = {            _year: 1994,            edition: 1        }        Object.defineProperty(book, "year", {            get: function () {                return this._year;            },            set: function (newValue) {                if (newValue > 1994) {                    this.edition += newValue - this._year;                    this._year = newValue;                }            }        });        book.year = 2016;        alert(book.edition);

以上代码创建了一个book对象,并给他定义了两个默认的属性:_year和edition。_year前面的下划线十一找那个常用的标记,用于表示只能通过通过对象方法访问的属性。而访问器属性year则包含一个getter函数和setter函数。getter函数返回_year值.这里不一定要同时指定getter和setter。只指定getter意味着属性是不能写,只指定setter意味着只写,无法获取属性值。

注意:支持ECMAScript 5的get,set方法浏览器只有IE9+(IE8部分实现)、FireFox 4+、Safari 5+、Opera 12+和Chrome,在这之前,要创建访问器属性,一般都使用两个非标准方法,_defineGetter_()和_defineSetter_(),如下代码是早期访问器属性的代码版本:

var book = {            _year: 1994,            edition: 1        }        book._defineGetter_("year", function () {            return this._year;        });        book._defineSetter_("year", function (newValue) {            if (newValue > 1994) {                this.edition += newValue - this._year;            }            this._year = newValue;        });        book.year = 2016;        alert(book.edition);

 

3、定义多个属性

在开发当中,当我们用到对象的时候,大多数情况下都会用到多个属性,所以ECMAScript 5又定义了一个Object.defineProperties()方法。利用这个方法可以通过描述符一次定义多个属性。

代码如下:

var book = {};    Object.defineProperties(book, {        _year: {            writable: true,            value: 2004        },        edition: {            writable: true,            value: 1        },        year: {            get: function () {                return this._year;            },            set: function (newValue) {                if (newValue > this._year) {                    this.edition += newValue - this._year;                }                this._year = newValue;            }        }    });    book.year = 2016;    alert(book.edition);

上面定义了两个数据属性(_year和edition)和一个访问器属性(year)。最终的对象和上一段代码定义的对象相同。唯一的区别是这里的属性都是在同一时间创建的!上一段代码中定义的对象,实在定义完数据属性之后又定义了一个访问器属性。

 

4、读取属性的特性

var book = {};    Object.defineProperties(book, {        _year: {            writable: true,            value: 2004        },        edition: {            writable: true,            value: 1        },        year: {            get: function () {                return this._year;            },            set: function (newValue) {                if (newValue > this._year) {                    this.edition += newValue - this._year;                }                this._year = newValue;            }        }    });    var attribute = Object.getOwnPropertyDescriptor(book, "_year"); /*获取数据属性_year对象注意:支持这个方法的浏览器有IE9+,fireFox4,Safari 5+,Opera,Chrome*/    alert(attribute.value);  /*输出数据属性_year的Value特性*/    alert(attribute.configurable); /*输出数据属性_year的configurable特性*/    alert(typeof attribute.get); /*输出数据属性的get的特性,但是数据属性被没有Get特性,只有访问器属性才有Get和Set特性,所以这里输出undefined*/    var attribute_two=Object.getOwnPropertyDescriptor(book,"year");    alert(attribute_two.get);  //输出访问器属性的get方法,get是指向getter函数的指针    alert(attribute_two.value);   //输出访问器属性year的value特性,但是访问器属性并没有value特性,value特性属于数据属性,所以输出undefined    alert(attribute_two.enumerable); //因为访问器属性year并不是在对象上直接定义的属性而是通过defineProperties()方法定义的属性,所以他的Enumerable特性为false,所以输出false;

 

转载于:https://www.cnblogs.com/GreenLeaves/p/5795706.html

你可能感兴趣的文章
Windows phone 8 学习笔记
查看>>
我的友情链接
查看>>
感悟贴2016-05-13
查看>>
参加婚礼
查看>>
Java重写equals方法和hashCode方法
查看>>
Spring ’14 Wave Update: Installing Dynamics CRM on Tablets for Windows 8.1
查看>>
MySQL 备份与恢复
查看>>
TEST
查看>>
PAT A1037
查看>>
(六)Oracle学习笔记—— 约束
查看>>
[Oracle]如何在Oracle中设置Event
查看>>
top.location.href和localtion.href有什么不同
查看>>
Gradle之module间依赖版本同步
查看>>
java springcloud版b2b2c社交电商spring cloud分布式微服务(十五)Springboot整合RabbitMQ...
查看>>
d3 v4实现饼状图,折线标注
查看>>
微软的云策略
查看>>
Valid Parentheses
查看>>
nginx 301跳转到带www域名方法rewrite(转)
查看>>
AIX 配置vncserver
查看>>
windows下Python 3.x图形图像处理库PIL的安装
查看>>