时间:2021-07-01 10:21:17 帮助过:4人阅读
{
name:'linjisong',
getName:function(){
return this.name;
}
}
这里最外层的一对大括号({})就表示这是一个对象字面量。另外,还有数组字面量的概念,在ECMAScript中,数组Array是一个继承了Object的对象实例,通过这个对象实例可以创建数组类型的实例,数组类型的实例也可以直接通过数组字面量来表示,方法如下:
代码如下:
[{
name:'linjisong',
age:29
},{
name:'oulinhai',
age:23
}]
这里一对中括号([])用于表示数组,这是一个包含了两个对象的数组。通过对象字面量和数组字面量,形成了难以想象的强大表现力,事实上,流行的JSON数据格式就是基于此。
3、创建对象实例
熟悉一般面向对象的朋友都知道,要创建一个类的实例,首先要定义这个类,然后用new关键字来创建这个类的实例(别和我说还可以使用反射,我的Java可学的不好……)。但是在ECMAScript中,根本没有类的概念,那么,对象实例要怎么创建呢?
在ECMAScript中尽管没有类,但是也有某种程度上类似的概念,承担这个角色的就是函数,可以通过new操作符和函数来创建对象实例——每一个对象实例都有一个用于创建这个实例的函数。最基本的函数就是Object(),它是用来创建最一般对象的函数,其它的诸如Number()函数,可以用来创建Number对象的实例,Boolean()函数,可以用来创建Boolean对象的实例:
代码如下:
var obj = new Object();//Object()函数,创建最一般的对象实例
var num = new Number(1);//Number()函数,创建Number对象的实例
var boo = new Boolean(true);//Boolean()函数,创建Boolean对象的实例
console.info(typeof num);//object
console.info(typeof Number(1));//number
console.info(typeof boo);//object
console.info(typeof Boolean(true));//boolean
(1)可以看到,要创建一个对象实例,首先需要有一个函数(称为构造函数),这个函数使用new调用时就是创建对象实例,不使用new时只是通常意义上的函数调用(如果这个函数在内部返回实例了,函数调用也可以创建对象)。
(2)所谓的内置对象实际上也就是内置了一些创建对象实例的函数而已,不同的函数创建不同的内置对象。
(3)关于要不要使用new操作符,我的建议是使用,如果不使用new操作符,有些情况下结果会出乎你的意料之外,像上例中的第5、7行,实际上并没有创建对象,而只是普通的函数调用,这个调用的作用就是转换数据类型。
(4)使用new创建对象实例时,如果调用构造函数不需要传入参数,也可以省略后面的函数调用操作符(()),当然,这种特性也不是什么值得宣扬的事情。
(5)如果需要创建自定义对象的实例,那么首先也需要定义一个构造函数,然后使用new操作符调用创建实例。这里需要注意,如果忘了new的话,可能会污染全局环境:
代码如下:输出全局的age,由于未定义,抛出异常
function Person(){//首先定义一个用于创建对象实例的(构造)函数
this.name = 'linjisong';
this.age = 29;
}
var person = new Person();//调用(构造)函数创建对象实例
console.info(person.age);//29
try{
console.info(age);//为了演示忘记使用new的情况,这里先
}catch(e){
console.info(e);//ReferenceError
}
var person2 = Person();//忘记使用new的情况下,只是普通的函数调用,由于函数没有返回,这里person2就是undefined了
console.info(person2);//undefined
console.info(age);//29,没有使用new,内部的this指向了全局作用域,因为可以在全局访问age了
要避免这种问题,可以修改一下构造函数:
代码如下:输出中得到验证:
function Person(){
if(this instanceof Person)
{
this.name = 'linjisong';
this.age = 29;
}else{
return new Person();
}
}
var person2 = Person();
console.info(person2.age);//29,可以访问person2的age了
console.info(age);//全局环境中没有age的定义了,抛出异常
这个构造函数首先判断this值是否为Person类型,如果不是,就在内部使用new调用,以确保返回的值一定是Person类型实例。这种方式使得重构构造函数成为了可能,也许Boolean()、Number()、String()在实现上就是使用了这种方式来区分是构造函数还是转换函数。如果你在调用Object()时省略new的话,结果也能返回对象,估计也是在后台做了类似处理,同样的情况还有本文后部分要讲的函数类型构造函数Function()。
(5)可能有人会问,既然有对象字面量,何必要用这么复杂的方式来创建对象实例呢,直接写对象字面量不就完了?用对象字面量创建对象实例,根本没有使用什么函数,看来,上面的“每一个对象实例都有一个用于创建这个实例的函数”的说法并不正确。
首先第一个问题,的确,可以使用对象字面量来创建函数,而且也非常简洁,这甚至也是我首先推荐的一种创建方式,但是用这种方式创建对象实例,只能创建单例的实例,对于需要创建多个相同类型的对象实例来说并不适用,然后第二个问题,用对象字面量创建对象,实际上并不是没有相应的构造函数,只是构造函数为Object(),使用对象字面量,后台可能不会去调用new Object(),但创建出的对象仍然有指向这个函数的属性,这可以从下面代码
代码如下:
var person = {};
console.info(person.constructor===Object);//true
这里的constructor是每个实例对象都有的一个属性,用于保存创建这个对象实例的函数,这就是下面要讲的。
4、对象属性和方法
每一种数据类型都有各自的共性,比如Number类型值都有可以和另外一个Number类型值相加的特性,同样,对象类型的实例也有一些相同的特性,这些特性就体现在它们都包含下面的属性和方法(方法实际上也是一种属性,只是属性的值类型是函数的话,我们也称之为方法):
类别 | 属性/方法 | 说明 |
属性 | constructor | 指向用于创建当前对象的函数 |
方法 | hasOwnProperty(propertyName) | 检查给定的属性是否在当前对象实例中 |
propertyIsEnumerable(propertyName) | 检查给定的属性是否能够是使用for-in语句来枚举 | |
isPrototype(object) | 检查传入的对象是否是另一个对象的原型 | |
toLocalString() | 返回对象的字符串表示,该字符串与执行环境的地区相对应 | |
toString() | 返回对象的字符串表示 | |
valueOf() | 返回对象的字符串、数值或布尔值表示,通常与toString()方法返回值相同 |
for(var propertyName in object){
if(object.hasOwnPorperty(propertyName)){
//循环处理
}
}