引用类型 - object
引用类型的值(对象)是引用类型的一个实例。在ECMAScript中,引用类型是一中数据结构,用于将数据和功能组织在一起。它也常被称做类,但这种称呼并不妥当。尽管ECMAScript从技术上讲是一门面向对象的语言,但它不具备传统的面向对象语言所支持的类和接口等基本结构。引用类型有时候也被称作对象定义,因为他们描述的是一类对象所具有的属性和方法。
如前所述,对象是某个特定引用类型的实例。新对象是使用new操作符后跟一个构造函数来创建的。构造函数本身就是一个函数,只不过该函数是出于创建新对象的目的而定义的。
Object类型
创建Object实例的方式有两种。第一种是使用new操作符后跟Object构造函数,如下所示:
var person=new Object(); person.name="Nicholas"; person.age=29;
另一种方式是使用对象字面量表示法。对象字面量是对象定义的一种简写形式,目的在于简化创建包含大量属性的对象的过程。下面这个例子就使用对象字面量语法定义了与前面那个例子相同的person对象:
var person={ name:"Nicholas"; age:29 }
在这个例子中,左边的花括号({)表示对象字面量的开始,因为它出现在了表达式上下文(expression context)中。ECMAScropt中的表达式上下文指的是能够返回一个值(表达式)。赋值操作符表示后面一个值,所以左花括号在这里表示一个表达式的开始。同样的花括号,如果出现在一个语句上下文(statement context)中,例如跟在if语句条件的后面,则表示一个语句块的开始。
然后,我们定义了name属性,之后是一个冒号,在后面是这个属性的值。在对象字面量中,使用逗号来分隔不同的属性,因此“Nicholas”后面是一个逗号,但是,在age属性的值29的后面不能添加逗号,因为age是这个对象的最后一个属性。在最后一个属性后面添加逗号,会在IE7及更早版本和Opera中导致错误。
在使用对象字面量语法时,属性名也可以使用字符串,如下面这个例子所示:
var person={ "name":"Nicholas", "age":29, 5:true }
这个例子会创建一个对象,包含三个属性:name,age和5。但这里的数值属性名会自动转换为字符串。
另外,使用对象字面量语法时,如果留空其花括号,则可以定义只包含默认属性和方法的对象,如下所示:
var person={}; person.name="Nicholas"; person.age=29;
这个例子与本节前面的例子是等价的,只不过看起来似乎有点奇怪。关于对象字面量语法,我们推荐只在考虑对象属性名的可读性时使用。
虽然可以使用恰面介绍的任何一种方法来定义对象,但开发人员更青睐对象字面量语法,因为这种语法要求的代码量最少,而且能够给人封装数据的感觉。实际上,对象字面量也是可以向函数传递大量可选参数的首选方式,例如:
function displayInfo(args){ var output=""; if(typeof args.name=="string"){ output+="Name:"+args.name+"\n"; } if(typeof args.age=="number"){ output+="Age:"+args.age+"\n" } alert(output); } displayInfo({ name:"Nicholas", age:29 }); displayInfo({ name:"Greg" });
在这个例子中,函数displayInfo()接受一个名为args参数。这个参数可能带有一个名为name或age的属性,也可能这两个属性都有或者都没有。在这个函数内部,我们通过typeof操作符来检测每个属性是否存在,然后再基于相应的属性来构造一条要显示的消息。然后,我们调用了两次这个函数,每次都使用一个对象字面量来指定不同的数据。这两个调用传递的参数虽然不同,但函数都能正常执行。
一般来说,访问对象属性时使用的都是点表示法,这也是很多面向对象语言中通用的语法。不过,在JavaScript也可以使用方括号中,如下面的例子所示:
alert(person["name"]);//Nicholas alert(person.name);//Nicholas
从能上看,这两种访问对象属性的方法没有任何区别。但方括号语法的主要优点是可以通过变量来访问属性,例如:
var propertyName="name"; alert(person[propertyName]);//Nicholas
如果属性名中包含会导致语法错误的字符,或者属性名使用的是关键字或保留字。也可以使用方括号表示法。例如:
person["first name"]="Nicholas";
由于“first name”中包含一个空格,所以不能使用点表示法来访问它。然而,属性名中是可以包含非字母非数字的,这时候就可以使用方括号表示法来访问它们。
通常,除非必须使用变量来访问属性,否则我们建议使用点表示法。