让前端飞饥人谷技术博客

面向对象01 理解对象和创建对象

2017-03-15  本文已影响0人  IT男的成长记录

理解对象和创建对象

1.理解对象

var person = new Object();
person.name = "Nicholas";
person.age = 29;
person.job = "Software Engineer";

person.sayName = function(){
    alert(this.name);

}
var person = {
    name: "Nicholas";
    age: 29;
    job: "Software Engineer";
    
    sayName: function(){
        alert(this.name);
    }
}

1.1 属性类型

ECMAScript中有两种 属性(property) :数据属性和访问器属性,用 特性(attribute) 来描述属性的各种特征。

数据属性:

数据属性包含一个数据值的位置,在这个位置可以读取和写入值。

数据属性有四个描述其行为的特性:

访问器属性

访问器属性不包含数据值,它们包含一对getter和setter函数(不是必须的)。在读取访问器属性时,会调用getter函数,在写入访问器属性时,会调用setter函数并传入新值。访问器属性有以下四个特性:

var book = {
    _year:2004;
    edition: 1;
};
Object.defineProperty(book,"year",{
    get:function(){
        return this._year;
    },
    set: function(newValue){
        if(newValue > 2004){
            this._year = newValue;
            this.edition+=newValue -2004;
        }
    }
};
book.year = 2005; //对访问器属性year进行写入,因此调用setter函数
alert(book.edition); //2

2 创建对象

Object构造函数和对象字面量

前面提到了早期创建对象的两种方法:Object构造函数对象字面量,但是这些方式有明显的缺点:使用同一个接口创建很多对象,会产生大量的重复代码。

工厂模式:

function createPerson(name, age, job){
    var o = new Object();
    o.name = name;
    o.age = age;
    o.job = job;
    o.sayName = function(){
        alert(this.name);
    }
    return o;
}
var person1 = create("Nicholas", 29, "Software Engineer");

缺点:

构造函数模式

ECMAScript中的构造函数可以用来创建特定类型的对象。

function Person(name, age, job){
    this.name=name;
    this.age=age;
    this.job=job;
    this.sayName=function(){
        alert(this.name);
    }
}
// 注意将Person当做构造函数,需要使用new操作符来创建新对象
var person1 = new Person("xin",22,"Software Engineer");
var person2 = new Person("wu",22,"Software Engineer");

在上面的例子中,person1和person2分别保存着一个不同的实例,但是这两个对象都有一个constructor(构造函数)属性,该属性执行Person。

检测对象类型:

alert(person1.constructor == Person); //true
alert(person1 instanceof Object); //true
alert(person1 instanceof Person); //true

创建自定义的构造函数意味着将来可以将它的事例标识为一种特定的类型

缺点:

function Person(name, age, job){
    this.name=name;
    this.age=age;
    this.job=job;
    this.sayName=sayName;//sayName属性设置为全局的sayName函数
}
function sayName(){
        alert(this.name);
}
var person1 = new Person("xin",22,"Software Engineer");
var person2 = new Person("wu",22,"Software Engineer");

原型模式

每个函数都有一个prototype(原型) 属性,这个属性是一个指针,指向一个对象,而这个对象的用途是 包含可以由特定类型的所有实例共享的属性和方法

function Person(){
}
Person.prototype.name = "Nicholas";
Person.prototype.age = 29;
Person.prototype.job = "Software Engineer";
Person.ptototype.sayName = function(){
    alert(this.name);
}
var person1 = new Person();
person1.sayName(); //"Nicholas"

var person2 = new Person();
person2.sayName();//"Nicholas"
//person1和person2的属性和方法是所有实例共享的
alert(person1.sayName == person2.sayName); //true
理解原型对象
image
原型属性[[Prototype]]的访问
alert(Person.prototype.isPrototypeOf(person1));//true
alert(Object.getPrototypeOf(person1) == Person.prototype); //true
alert(Object.getPrototypeOf(person1).name);//Nicholas

person1.hasOwnProperty("name"); //false

alert( !person1.hasOwnProperty(name) && name in person1); //true
更简单的原型语法:

前面的例子中每添加一个属性和方法就要敲一遍Person.prototype。为了减少不必要的输入,更常见的方法是 用一个包含所有属性和方法的对象字面量来重写整个原型对象

function Person(){
}
Person.prototype = {
    name: "Nicholas",
    age: 29,
    job: "Software Engineer",
    sayName: function(){
        alert(this.name);
    }
}

注意
这里使用的语法,本质上完全重写了默认的prototype对象,因此constructor属性也就变成了新对象的constructor属性(指向Object构造函数),不再指向Person函数。使用instanceof操作符还能返回正确的结果,但是通过constructor已经无法确定对象的类型了。

var friend = new Person();
alert(friend.instanceof Person);//true
alert(friend.constructor == Person);//false

如果constructor属性很重要,可以将其设为适当的值

//方法一,但是会使constructor属性的[[Enumerable]]特性变为true
function Person(){
}
Person.prototype = {
    constructor:Person,
    name:"Nicholas",
    age:29,
    job:"Software Engineer",
    sayName:function(){
        alert(this.name);
    }
}
function Person(){
}
Person.prototype = {
    name:"Nicholas",
    job:"Software Engineer",
    age:29,
    sayName:function(){
        alert(this.name);
    }
}
Object.defindProperty(Person.prototype, "constructor",{ 
    enumerable: false,
    value: Person
});
原生对象的原型:

所有原生类型(Object,Array,String,等等)都在其构造函数的原型上定义了方法。

原型对象的问题:
function Person(){
}
Person.prototype = {
    constructor: Person,
    name: "Nicholas",
    age: 29,
    job: "Software Engineer",
    friends: ["Sheldon","Court"],
    sayName: function(){
        alert(this.name);
    }
};
var person1 = new Person();
var person2 = new Person();

person1.friends.push("van");
alert(person1.friends); //sheldon,court,van
alert(person2.friends); //sheldon,court,van
alert(person1.friends === person2.friends);//true

出现上述问题的原因在于:person1和person2的friends属性共享一个数组。

组合使用构造函数模式和原型模式

构造函数模式用来定义实例属性,原型模式用来定义方法和共享属性。结果每个实例都会有自己的一份实例属性的副本,但同时又共享着对方法的引用,最大限度的节省了内存。

function Person(name,age,job){
    this.name = name;
    this.age = age;
    this.job = job;
    this.friends = ["sheldon","mary"];
}
Person.prototype = {
    constructor: Person,
    sayName: function(){
        alert(this.name);
    }
}

稳妥构造函数模式

function Person(name,age,job){
    var o = new Object();
    o.sayName = function(){
        alert(name);
    }
    return o;
}
var friend = Person("Nicholas",29,"Software Engineer");
friend.sayName();

即使有其他代码会给这个对象添加方法或数据成员,但也不可能有别的方法访问传入到构造函数中的原始数据(name,job,age)。稳妥构造函数模式提供的这种安全性,使得它非常适合在某些安全环境中执行。

使用稳妥构造函数模式创建的对象与构造函数之间没什么关系,因此instanceof操作符对这种对象没有什么意义

上一篇下一篇

猜你喜欢

热点阅读