zoukankan      html  css  js  c++  java
  • 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;
  • 相关阅读:
    在项目中运用到的导航高亮
    【转载】IE8 inlineblock容器不撑开问题(利用重绘解决)
    我的博客正式开通
    【转载】响应式网页设计的9条基本原则
    一款不错的在线SVG制作工具
    【转载】前端不为人知的一面前端冷知识集锦
    11.3 Daily Scrum
    11.11 Daily Scrum
    11.7 Daily Scrum(周末暂停两天Daily Scrum)
    11.12 Daily Scrum(保存草稿后忘了发布·····)
  • 原文地址:https://www.cnblogs.com/GreenLeaves/p/5795706.html
Copyright © 2011-2022 走看看