JavaScript之面向对象的概念,对象属性和对象属性的特性简介

时间:2022-04-24
本文章向大家介绍JavaScript之面向对象的概念,对象属性和对象属性的特性简介,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。

  一、大家都知道,面向对象语言有一个标志,那就是他们都有类的概念,通过类我们可以创建任意多个具有相同属性和方法的对象。但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;