TDD测试驱动的javascript开发(3) ------ javascript的继承
说起面向对象,人们就会想到继承,常见的继承分为2种:接口继承和实现继承。接口继承只继承方法签名,实现继承则继承实际的方法。
由于函数没有签名,在ECMAScript中无法实现接口继承,只支持实现继承。
1. 原型链
1.1 原型链将作为实现继承的主要方法,基本思想是利用原型让一个引用类型继承另一个引用类型的属性和方法。
构造函数---原型---实例 之间的关系:
每一个构造函数都有一个原型对象,原型对象包含一个指向构造函数的指针,而实例都包含一个指向原型对象的内部指针。
function SuperType(name) {this.name = name;this.colors = ['red','green'];}SuperType.prototype.sayName = function() {return this.name;};/** * 1.创建父类型的一个副本 * 2.弥补创建副本时所丢失的constructor属性 * 3.将新的副本赋值给子类型的原型 * @param subType * @param superType */function inheritPrototype(subType,superType) {var object = Object.create(superType.prototype);//创建对象object.constructor = subType; //增强对象subType.prototype = object;//指定对象}function SubType(name,age) {SuperType.call(this,name); //调用构造函数this.age = age;}inheritPrototype(SubType,SuperType);SubType.prototype.sayAge = function() {return this.age;};var subInstance1 = new SubType('feng',28); subInstance1.colors.push('yellow');var subInstance2 = new SubType('tong',25);TestCase("test parasitic extends",{"test subInstance1 name property should be feng" : function() {assertEquals("feng",subInstance1.name); },"test subInstance2 name property should be tong" : function() {assertEquals('tong',subInstance2.name); },"test subInstance1 colors property length should be 3" : function() {assertEquals(3,subInstance1.colors.length); },"test subInstance2 colors property length should be 2" : function() {assertEquals(2,subInstance2.colors.length); },"test subInstance1 sayAge method should be return 28" : function() {assertEquals(28,subInstance1.sayAge()); },"test subInstance2 sayAge method should be return 25" : function() {assertEquals(25,subInstance2.sayAge()); },"test subInstance1 should be instanceof SuperType" : function() {assertInstanceOf(SuperType,subInstance1); }});1.8总结
1.8.1 ECMAScript 支持面向对象编程,但不使用类或者接口,对象可以在代码执行过程中创建和增强。
1.8.2 创建对象的几种方式:
构造函数模式:可以创建自定义的引用类型,使用new操作符
缺点:在每个实例上都要重新创建,无法复用,包括函数
优点:与对象的松耦合
适用场合:当属性或者方法不做共享属性或者方法的时候(比如引用类型的属性),不建议单独使用。
原型模式:使用构造函数的prototype属性来指定那些共享的属性和方法
优点:所有成员都可以共享属性和方法
缺点:没有私有属性值
适用场合:所以的属性和方法都可以被共享的时候,不建议单独使用
组合使用构造函数和原型模式:使用构造函数定义实例属性,使用原型定义共享属性和方法。
优点:解决了原型模式共享引用类型的属性的问题,也解决了构造函数不能共享属性的问题。
缺点:实现继承的时候,将会2次调用父类型的构造函数。性能问题。
使用场合:使用最广泛的定义引用类型的一种默认模式。
动态原型模式:保持了构造函数和原型的优点,把所有的信息封装在构造函数内,在有必要的情况下进行初始化原型。
稳妥构造函数模式:适合在安全环境中使用。
1.8.3 javascript的继承:
javascript主要通过原型链事实现继承。
原型链的继承:原型链的构建是通过将一个类型的实例赋值给另一个构造函数的原型实现的。
缺点:对象实例共享所有继承的属性和方法,因此不宜单独使用。
借用构造函数继承:在子类构造函数的内部调用超类型的构造函数。call() apply()
缺点:没有函数的复用,不建议单独使用
组合继承:使用原型链继承共享的属性和方法,而通过借用构造函数继承实例属性。
缺点:将会调用两次构造函数,会有性能问题
原型式继承:可以在不必预定义构造函数的情况下实现继承,其本质是执行对给定对象的浅复制,而复制的副本还可以得到进一步的改造。
优点:在没有必要创建构造函数,只是让一个对象与另一个对象保持类似的情况下,原型式继承OK.
缺点:共享属性和方法
寄生式继承:与原型式继承非常相似,基于对象获某些信息创建一个对象,然后增强对象,最后返回对象
优点:解决组合继承多次调用父类的构造函数而导致低效率问题。(可以与组合模式一起使用)
缺点:使用寄生式继承来为对象添加函数,会由于不能做到函数复用而降低效率,与构造函数模式类似。
寄生组合式继承:集寄生式继承和组合继承的优点于一身,是实现基于类型继承的最有效方式。