ECMAScript 中没有类的概念。
ECMA-262把对象定义为:“无序属性的集合,其属性可以包含基本值,对象或者函数。”,对象是一组没有特定顺序的值。可以把对象想象成为散列表,无非就是一组名值对,其中值就是数据或者函数。
每个对象都是基于一个引用类型(原生类型/自定义类型)创建的。
创建对象的几种方式:
1. 创建一个object实例,然后为它添加属性和方法
var person = new Object(); person.name = "da huang"; person.age = 18; person.job = "software developer"; person.sayName = function (){ alert (this.name); } person.sayName();缺点:使用同一个接口创建很多对象,会产生大量的重复代码。
2. 工厂模式
考虑到ECMAScript 无法创建类,故用函数来封装以特定接口创建对象的细节
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 = createPerson("Li Lei", 19, "PHD"); var person2 = createPerson("Da Huang", 18, "software developer"); person1.sayName(); person2.sayName();特点:工厂模式虽然解决了创建多个相似对象的问题,但是,却 没有解决对象识别的问题(即怎样知道一个对象的类型)。
ECMAScript中的构造函数可以用来创建特定类型的对象。像array和object这样的原生构造函数,会自动出现在执行环境中。此外,也可以创建自定义的构造函数,从而定义自定义对象类型的属性和方法。
// 构造函数 function Person(name, age, job) { this.name = name; this.age = age; this.job = job; this.sayName = function () { alert(this.name); } } var person1 = new Person("Li Lei", 19, "PHD"); var person2 = new Person("Da Huang", 18, "software developer"); person1.sayName(); person2.sayName(); alert(person1 instanceof Person);// true alert(person1 instanceof Object);// true
上面的person1和person2都有contructor属性,constructor属性最初是用来标志对象类型的,但是,提到监测对象,还是instanceof操作符更靠谱一点。person1既是Person实例,又是Object对象,是因为所有对象均继承自Object。
将构造函数作为函数
构造函数与其他函数的唯一区别,在于调用方式不同。任何函数只要通过new来调用,那它就可以作为构造函数;任何函数如果不通过new操作符调用,那它跟普通函数没什么两样。
//构造函数方式调用 var person1 = new Person("Li Lei", 19, "PHD"); person1.sayName(); //普通函数方式调用 Person("Yuan Jie", 18, "software developer"); window.sayName(); //在另一个对象中调用 var o = new Object(); Person.call(o,"Li Lei", 19, "software developer"); o.sayName();不使用new操作符调用Person(),属性和方法都被添加到了window对象。当在全局作用域调用一个函数时,this对象总是指向Global对象(浏览器中是window对象),因此,在调用完函数后,可以通过window对象来调用sayName(),也可以通过call()(apply())在某个特殊对象的作用域中调用Person() 函数,这里在o的作用域中调用,因此调用后,o就拥有了所有属性和方法。
4. 原型模式
// 原型模式 function Person(){ } Person.prototype.name = "Da Huang"; Person.prototype.age = 18; Person.prototype.sayName = function(){ alert(this.name); } var person1 = new Person(); person1.sayName();// Da huang 来自原型 person1.name = "Li Lei"; person1.sayName(); //Li Lei 来自实例 person1.name = null; if (person1.name == null) ; alert("person1.name == null");// person1.name == null 来自实例 delete person1.name; person1.sayName(); // Da Huang 来自原型 var person2 = new Person(); person2.sayName(); // Da Huang 来自原型