对象是一个包含相关数据和方法的集合(通常由一些变量和函数组成,我们称之为对象里面的属性和方法),让我们通过一个例子来了解它们。
首先, 将oojs.html文件复制到本地. 此文件包含非常少 — 一个供我们写源代码的<script>标签, 一个供我们输入示例指令的<input>标签,当页面被渲染时, 一些变量定义, 一个输出任何输入到<input>的内容输出到<p>标签的函数。我们用这个文件做为基础探索对象的基础语法。
如同Javascript中的很多东西一样,创建一个对象通常先定义初始化变量。 尝试在您已有的文件中JavaScript代码下面输入以下内容, 保存刷新页面:
varperson={};
如果你在浏览器控制台输入person,然后按下Enter(确认)键,你会得到如下结果:
[object Object]
恭喜, 你刚创建了你的第一个对象. 干的漂亮! 但这是一个空对象,所以我们做不了更多的事情。像下面一样更新下我们的对象:
var person = {
name : ['Bob', 'Smith'],
age : 32,
gender : 'male',
interests : ['music', 'skiing'],
bio : function() {
alert(this.name[0] + ' ' + this.name[1] + ' is ' + this.age + ' years old. He likes ' + this.interests[0] + ' and ' + this.interests[1] + '.');
},
greeting: function() {
alert('Hi! I\'m ' + this.name[0] + '.');
}
};
保存刷新后, 尝试在你的浏览器控制台输入下面的内容:
person.name[0]
person.age
person.interests[1]
person.bio()
person.greeting()
现在在你的对象里得到了一些数据和功能(functionality),现在可以通过简单的语法访问他们了!
Note:如果做上面的东西遇到了麻烦,尝试拿你的代码与我们的版本做对比——对比oojs-finished.html (也可以 看实际效果)。一个对于初学者很常见的错误是在最后一个成员后面多了一个逗号,这会引发错误。
所以发生了什么?一个对象由许多的成员组成,每一个成员都拥有一个名字(像上面的name、age),和一个值(如['Bob', 'Smith']、32)。每一个名字/值(name/value)对被逗号分隔开,并且名字和值之间由冒号(:)分隔,语法规则如下所示:
var objectName = {
member1Name : member1Value,
member2Name : member2Value,
member3Name : member3Value
}
对象成员的值可以是任意的,在我们的person对象里有字符串(string),数字(number),两个数组(array),两个函数(function)。前4个成员是资料项目,被称为对象的属性(property),后两个成员是函数,允许对象对资料做一些操作,被称为对象的方法(method)
一个如上所示的对象被称之为对象的字面量(literal)——手动的写出对象的内容来创建一个对象。不同于从类实例化一个对象,我们会在后面学习这种方式。
当你想要传输一些有结构和关联的资料时常见的方式是使用字面量来创建一个对象,举例来说,发起一个请求到服务器以存储一些数据到数据库,发送一个对象要比分别发送这些数据更有效率,而且比起数组更为易用,因为你使用名字(name)来标识这些资料。
在上面的例子中,你使用了点表示法(dot notation)来访问对象的属性和方法。对象的名字表现为一个命名空间(namespace),它必须写在第一位——当你想访问对象内部的属性或方法时,然后是一个点(.),紧接着是你想要访问的项目,标识可以是简单属性的名字(name),或者是数组属性的一个子元素,又或者是对象的方法调用。如下所示:
person.age
person.interests[1]
person.bio()
可以用一个对象来做另一个对象成员的值。例如将name成员
name : ['Bob', 'Smith'],
改成
name : {
first : 'Bob',
last : 'Smith'
},
这样,我们实际上创建了一个子命名空间,听起来有点复杂,但用起来很简单,你只需要链式的再使用一次点表示法,像这样:
person.name.first
person.name.last
注意:你需要改变你之前的代码,从
name[0]
name[1]
改成
name.first
name.last
否则,你的方法不再有效。
另外一种访问属性的方式是使用括号表示法(bracket notation),替代这样的代码
person.age
person.name.first
使用如下所示的代码:
person['age']
person['name']['first']
这看起来很像访问一个数组的元素,从根本上来说是一回事儿,你使用了关联了值的名字,而不是索引去选择元素。难怪对象有时被称之为关联数组(associative array)了——对象做了字符串到值的映射,而数组做的是数字到值的映射。
目前我们仅仅看到了如何访问对象的成员,而你其实也可以设置对象成员的值,通过声明你要设置的成员,像这样:
person.age = 45
person['name']['last'] = 'Cratchit'
尝试这些代码,然后再查看这些成员是否已经被改变了
person.age
person['name']['last']
设置成员并不意味着你只能更新已经存在的属性的值,你完全可以创建新的成员,尝试以下代码:
person['eyes'] = 'hazel'
person.farewell = function() { alert("Bye everybody!") }
现在你可以测试你新创建的成员
person['eyes']
person.farewell()
括号表示法一个有用的地方是它不仅可以动态的去设置对象成员的值,还可以动态的去设置成员的名字。
比如说,我们想让用户能够在他们的数据里存储自己定义的值类型,通过两个input框来输入成员的名字和值,通过以下代码获取用户输入的值:
var myDataName = nameInput.value
var myDataValue = nameValue.value
我们可以这样把这个新的成员的名字和值加到person对象里:
person[myDataName] = myDataValue
为了测试这个功能,尝试在你的代码里添加以下几行,就在person对象的右花括号的下面:
var myDataName = 'height'
var myDataValue = '1.75m'
person[myDataName] = myDataValue
现在,保存并刷新,在输入框里输入以下代码:
person.height
这是使用点表示法无法做到的,点表示法只能接受字面量的成员的名字,不接受变量作为名字。
你也许在我们的方法里注意到了一些奇怪的地方,看这个例子:
greeting: function() {
alert('Hi! I\'m ' + this.name.first + '.');
}
你也许想知道"this"是什么,关键字"this"指向了当前代码运行时的对象( 原文:the current object the code is being written inside )——这里即指person对象,为什么不直接写person呢?当你学到下一篇Object-oriented JavaScript for beginners文章时,我们开始使用构造器(constructor)时,"this"是非常有用的——它保证了当代码的上下文(context)改变时变量的值的正确性(比如:不同的person对象拥有不同的name这个属性,很明显greeting这个方法需要使用的是它们自己的name)。
让我们以两个简单的person对象来说明:
var person1 = {
name : 'Chris',
greeting: function() {
alert('Hi! I\'m ' + this.name + '.');
}
}
var person2 = {
name : 'Brian',
greeting: function() {
alert('Hi! I\'m ' + this.name + '.');
}
}
在这里,person1.greeting()会输出:"Hi! I'm Chris.";person2.greeting()会输出:"Hi! I'm Brain.",即使greeting这个方法的代码是一样的。就像我们之前说的,this 指向了代码所在的对象(其实代码运行时所在的对象)。在字面量的对象里this看起来不是很有用,但是当你动态创建一个对象(例如使用构造器)时它是非常有用的,之后你会更清楚它的用途。
当你使用过这些例子之后,你可能会发现你对点表示法并不陌生,这是因为我们在这个课程里一直在使用它,每次我们学习的示例使用浏览器内建的API和JavaScript的一些对象时,我们就在使用对象,因为,这些功能是由跟我们所看到的对象同样的结构来构建的,虽然比我们自己定义的要复杂许多。
所以当我们这样使用字符串的方法时:
myString.split(',');
你正在使用一个字符串实例上可用的方法,你随时都可以在代码里使用字面量创建一个字符串,字符串会自动的被创建为字符串(String)的实例,因此会有一些常见的方法和属性可用。
当你这样访问document对象时:
var myDiv = document.createElement('div');
var myVideo = document.querySelector('video');
你正在使用Document实例上可用的方法。每个页面在加载完毕后,会有一个Document的实例被创建,叫做document,它代表了整个页面的结构,内容和一些功能,比如页面的URL。同样的,这意味document有一些可用的方法和属性。
这同样适用许多其他内建的对象或API,你使用过有—— Array,Math, 等。
请注意内建的对象或API不会总是自动地创建对象的实例,举例来说,这个 Notifications API——允许浏览器发起系统通知,需要你为每一个你想发起的通知都使用构造器进行实例化。尝试在JavaScript终端里输入以下代码
var myNotification = new Notification('Hello!');
我们会在之后的文章里学习到构造器。
Note: 这样来理解对象之间通过消息传递来通信是很有用的——当一个对象想要另一个执行某种动作时,它通常会通过那个对象的方法给其发送一些信息,并且等待回应,即我们所知的返回值。