JavaScript面向对象的方法学习
时间:2007-03-09 来源:xiaobian
URL:http://www.javaeye.com/topic/53537?page=1
ECMAScript可以识别两种类型的对象,一种叫做Native Object属于语言范畴;一种叫做Host Object,由运行环境提供例如document对象,
Dom Node等
Native objects是一种松散的结构并且可以动态的增加属性(property),所有的属性都有一个名字和一个值,这个值可以是另一个对象的引用
或者是内建的数据类型(String, Number, Boolean, Null 或者 Undefined)
下面的这个简单的例子描述了一个javascript对象是如何设置一个属性的值和如何读取属性的值的。
赋值操作
一个对象的属性的创建非常简单,直接通过赋值操作就可以完成属性的创建。
- var objectRef = new Object(); //create a generic javascript object.
一个名为testNumber的属性可以这样创建。
代码
- objectRef.testNumber = 5;
- /* - or:- */
- objectRef["testNumber"] = 5;
如果复制的属性名称已经存在那么不会再次创建这个属性,赋值操作仅仅是重新设置属性的值
代码
- objectRef.testNumber = 8;
- /* - or:- */
- objectRef["testNumber"] = 8;
js对象的原型(prototype)自己本身也可以是对象,也可以有属性(property),对于js对象的(prototype)的赋值操作跟普通对象属性的创建
没什么不同。
取值操作
在取值操作中property和prototype就不一样了,先看最简单的property取值操作。
代码
- /*为一个对象的属性赋值,如果这个对象没有这个属性,那么在赋值操作后,这个对象就有这个属性了 */
- objectRef.testNumber = 8;
- /* 读出这个属性的值 */
- var val = objectRef.testNumber;
- /* 现在val 就得到了刚才赋予objectRef的值8了*/
prototype揭密
但是所有的对象都可以有prototypes, prototypes自己也是对象,那么他也可以有prototypes,这样循环下去就形成了一个prototype链,
这个链当他遇到链中队形的prototype是null时中止。(Object的默认的prototype是null)
代码
- var objectRef = new Object(); //create a generic javascript object.
创建一个新的js对象,这时这个对象的prototype是Null,所以objectRef的prototype链只包含一个对象Object.prototype
我们在看下面的代码
代码
- /* 构建MyObject1这个类型的构造函数
- MyObject1 - type.
- */
- function MyObject1(formalParameter){
- /* 为者对象创建一个属性名字叫testNumber
- */
- this.testNumber = formalParameter;
- }
- /* 构建MyObject2这个类型的构造函数
- MyObject2 - type:-
- */
- function MyObject2(formalParameter){
- /* 为者对象创建一个属性名字叫testString*/
- this.testString = formalParameter;
- }
- /* 下一步的操作会用MyObject1对象替换掉MyObject2默认的prototype属性*/
- MyObject2.prototype = new MyObject1( 8 );
- /* 最后我们创建MyObject2类型的一个对象*/
- var objectRef = new MyObject2( "String_Value" );
objectRef这个MyObject2类型的对象有一个prototype的链,链中的第一个对象是MyObject1对象,MyObject1对象也有prototype,
这个prototype是Object默认的prototype,Object.prototype的prototype是null,至此这条prototype链结束。
当一个取值操作发生时,objectRef 的整个prototype链就开始工作
代码
- var val = objectRef.testString;
objectRef这个对象的有一个属性叫做testString,那么这句代码会把testString的值赋给val
代码
- var val = objectRef.testNumber;
在objectRef这个对象里并没有testNumber这个属性,但是val却的到了值8,而不是undefine,这是因为解释器在没有在当前对象找到要找
的属性后,就会去检查这个对象的prototype,objectRef的prototype是MyObject1对象,这个对象有testNumber这个属性,所以val得到8这个值。
代码
- var val = objectRef.toString;
现在val是个function的引用,这个function是Object.prototype的property,由于MyObject1和MyObject2都没有定义toString这个property
所以Object.prototype返回。
代码
- var val = objectRef.madeUpProperty;
最后val是undefined,因为MyObject1和MyObject2,还有Object都没有定义madeUpProperty这个property,所以得到的是undefine.
读操作会读取在obj自己和prototype 链上发现的第一个同名属性值
写操作会为obj对象本身创建一个同名属性(如果这个属性名不存在
这就意味着objectRef.testNumber = 3会在objectRef对象上创建一个property,名字是testNumber,当下一次在要读取testNumber时
propertype链就不会工作,仅仅会得到objectRef的property 3,而MyObject1的testNumber属性并不会被修改。下面的代码可以验证
- /* 构建MyObject1这个类型的构造函数
- MyObject1 - type.
- */
- function MyObject1(formalParameter){
- /* 为者对象创建一个属性名字叫testNumber
- */
- this.testNumber = formalParameter;
- }
- /* 构建MyObject2这个类型的构造函数
- MyObject2 - type:-
- */
- function MyObject2(formalParameter){
- /* 为者对象创建一个属性名字叫testString*/
- this.testString = formalParameter;
- }
- /* 下一步的操作会用MyObject1对象替换掉MyObject2默认的prototype属性*/
- var obj1 = new MyObject1( 8 );
- MyObject2.prototype = obj1;
- /* 最后我们创建MyObject2类型的一个对象*/
- var objectRef = new MyObject2( "String_Value" );
- alert(objectRef.testNumber);
- objectRef.testNumber = 5;
- alert(objectRef.testNumber);
- alert(obj1.testNumber);
Class - 类创建
Class类实现了在JavaScript中声明一个新的类, 并通过构造函数实例化这个类的机制。通过使用Class.create()方法, 你实际上声明了一个新的类, 并定义了一个initialize()方法作为构造函数, 一旦你在这个声明的类的prototype中实现了改该方法, 你就可以使用new操作符来创建并实例化一个类。
Knowledge Prepare - 知识准备
在JavaScript中, 当你定义了一个新的函数, 你实际上声明了一个新的类, 而这个函数本身就相当于类的构造函数。 下面的代码向你展示了两种不同的方式来创建一个新的Person类, 而Person.prototype的定义也紧跟在函数定义之后。
var Person = function(name) { // 一个匿名函数, 并将这个函数赋值给一个Person变量, 此时Person成为一个类
this.name = name;
}
function Person(name) { // 直接定义一个叫做Person的函数表示Person类
this.name = name;
}
Person.prototype = { // 定义Person的prototype域
printName: function() { // 定义一个print函数
alert(this.name);
}
}
当你通过函数的方式声明了一个类之后, 你就可以通过new操作符来实例化这个类。这样, 你就可以调用类的成员函数来完成你的逻辑。
var person = new Person("Joe Smith"); // 使用new操作符来新建一个Person的实例, 并赋给变量person
person.printName(); // person就可以看作是一个实例的引用(reference), 所以可以通过这个引用来调用Person类中的成员函数
我们来总结一下创建一个新的类的实例的整个流程和步骤:
1. 通过定义一个函数的方式(匿名或者实名)来声明一个新的类.
2. 如果有必要, 定义这个新的类的prototype域.
3. 使用new操作符紧跟你所定义的函数来创建一个新的类的实例. 一旦JavaScript编译器碰到了new操作符, 它实际上创建了一个空的类实例变量.
4. 将所有这个类的prototype域中的属性与方法复制到这个新的实例中, 并将其成员函数中所有的this指针指向这个新创建的实例.
5. 接下来, 执行紧跟在new操作符后面的那个函数.
6. 当你执行这个函数时, 如果你试图对一个不存在的属性进行赋值, JavaScript编译器将自动为你在这个实例范围内新创建这个属性.
7. 函数执行完毕后, 将这个初始化完成的实例返回.
在Prototype中, 使用Class对象, 你可以以一个比较简单的方式来声明一个新的对象。通过使用Class.create(), prototype为你创建了一个默认的构造函数initialize(), 一旦你实现这一函数, 就可以以一个类似Java中构造函数的方式来创建一个新的类的实例。
Source View - 源码解析
var Class = { // 全局静态类, 用于声明一个新的类并提供构造函数支持
create: function() {
return function() { // 返回一个函数, 代表着这个新声明的类的构造函数
// 一个命名为initialize的函数将被这个类实现作为类的构造函数
this.initialize.apply(this, arguments);// initialize函数将在你实例化一个变量的时候被调用执行(即上面7个步骤中的第5步)
}
}
}
Field & Function Reference - 属性方法一览
Class ( 静态 )
Method / Property | Kind | Arguments | Description |
---|---|---|---|
create() | 静态方法 | / | 用于声明一个新的类并提供了一个名为initialize构造函数支持 |
Analysis & Usage - 分析与使用
通过Class类, 你可以很容易地使用构造函数的方式创建一个新的类, 这对于Java程序员来说或许更加容易被接受。下面我们列出了Java和JavaScript各自声明和创建一个新的类的代码对比, 我们可以看到, 他们是如此相似:
var Person = Class.create(); // 类的声明 |public class Person { // 类的声明
Person.prototype = { | private String name;
initialize: function(name) { // 构造函数 | public Person(String name){ // 构造函数
this.name = name; | this.name = name;
} | }
printName: function() { // 成员函数 | public void printName(){ // 成员函数
alert(this.name); | System.out.println(name);
} | }
} |}
var person = new Person("Joe Smith");// 创建实例 |Person person = new Person("Joe Smith");// 创建实例
person.printName(); // 函数调用 |person.printName(); // 函数调用