说起面向对象,人们就会想到继承,常见的继承分为2种:接口继承和实现继承。接口继承只继承方法签名,实现继承则继承实际的方法。

由于函数没有签名,在ECMAScript中无法实现接口继承,只支持实现继承。

1. 原型链

1.1 原型链将作为实现继承的主要方法,基本思想是利用原型让一个引用类型继承另一个引用类型的属性和方法。

构造函数---原型---实例 之间的关系:

每一个构造函数都有一个原型对象,原型对象包含一个指向构造函数的指针,而实例都包含一个指向原型对象的内部指针。

function SuperType() {
this.property = true;
} SuperType.prototype.getSuperValue = function() {
return this.property;
}; function SubType() {
this.subproperty = false;
} SubType.prototype = new SuperType(); //通过原型链实现继承 SubType.prototype.getSubValue = function() {
return this.subproperty;
}; var subInstance = new SubType(); var superInstance = new SuperType(); TestCase("test extends",{ "test superInstance property should be true" : function() {
assertEquals(true,superInstance.property);
},
"test superInstance getSuperValue() should be return true" : function() {
assertEquals(true,superInstance.getSuperValue());
},
"test subInstance property should be false" : function() {
assertEquals(false,subInstance.subproperty);
},
"test subInstance could visit super method " : function() {
assertEquals(true,subInstance.getSuperValue()); //SubType继承SuperType,并调用父类的方法
}
});

注:要区分开父类和子类的属性名称,否则子类的属性将会覆盖父类的同名属性值:看如下代码:

function SubType() {
this.property = false;
}
SubType.prototype.getSubValue = function() {
return this.property;
};
function SuperType() {
this.property = true;
} SuperType.prototype.getSuperValue = function() {
return this.property;
}; SubType.prototype = new SuperType(); //通过原型链实现继承 var subInstance = new SubType(); var superInstance = new SuperType(); TestCase("test extends",{ "test superInstance property should be true" : function() {
assertEquals(true,superInstance.property); //父类的property值为true
},
"test superInstance getSuperValue() should be return true" : function() {
assertEquals(true,superInstance.getSuperValue()); //superInstance调用方法
},
"test subInstance property should be false" : function() {
assertEquals(false,subInstance.property); //子类的property属值为false
},
"test subInstance could visit super method " : function() {
assertEquals(false,subInstance.getSuperValue()); //SubType继承SuperType,并调用父类的方法,可以属性被覆盖了,返回false
}
});

续:当然,如果我们不要求对属性值进行初始化的时候,就不必考虑这个问题,我们会采用上一章讲的构造函数模式+原型模式来创建类和实现继承关系。

当以读取模式访问一个实例属性时,首先会在实例中搜索该属性,如果没有找到该属性,则继续在实例的原型中寻找。在通过原型链实现继承的情况下,会继续沿着原型链继续向上

subExtends.getSuperValue()

首先在实例中查找,然后在SubType.prototype,最后在SuperType.prototype中找到。

补充: 所有函数的默认原型都是Object的实例,因此默认原型都会包含一个内部指针,指向Object.prototype,这也正是所有自定义类型都会继承toString()、valueOf()等默认方法的原因。

1.2 确定原型和实例的关系

方法一:使用instanceof 操作符     ----     只要该实例是原型链中出现过的构造函数,结果就会返回true

function SuperType() {
this.property = true;
} SuperType.prototype.getSuperValue = function() {
return this.property;
}; function SubType() {
this.subproperty = false;
} SubType.prototype = new SuperType(); //通过原型链实现继承 SubType.prototype.getSubValue = function() {
return this.subproperty;
}; var subInstance = new SubType(); TestCase("test extends",{ "test subInstance should instanceof Object" : function() {
assertInstanceOf(Object,subInstance);
},
"test subInstance should instanceof SuperType " : function() {
assertInstanceOf(SuperType,subInstance);
},
"test subInstance should instanceof SubType " : function() {
assertInstanceOf(SubType,subInstance);
}
});

方法二:使用isPrototypeOf()方法     ----    只要原型链中出现过该原型,都可以说是该原型链所派生的实例的原型,结果会返回true

function SuperType() {
this.property = true;
} SuperType.prototype.getSuperValue = function() {
return this.property;
}; function SubType() {
this.subproperty = false;
} SubType.prototype = new SuperType(); //通过原型链实现继承 SubType.prototype.getSubValue = function() {
return this.subproperty;
}; var subInstance = new SubType(); TestCase("test extends",{ "test subInstance isPrototypeOf Object" : function() {
assertEquals(true,Object.prototype.isPrototypeOf(subInstance));
},
"test subInstance isPrototypeOf SuperType " : function() {
assertEquals(true,SuperType.prototype.isPrototypeOf(subInstance));
},
"test subInstance isPrototypeOf SubType " : function() {
assertEquals(true,SubType.prototype.isPrototypeOf(subInstance));
}
});

注:在实践中,我们很少会单独的使用原型链,因为它存在两个问题:

一、引用类型值的原型:包含引用类型值的原型属性会被所有实例共享,这也正是为什么要在构造函数中定义属性,而不在原型中定义属性的原因

二、创建子类型的实例时,不能向超类型的构造函数中传递参数。

1.3  借用构造函数(伪造对象  ----  经典继承)

原理: 在子类型构造函数的内部调用超类型的构造函数

function SuperType(name) {
this.name = name;
this.friends = ['tong','feng'];
} function SubType(name,age) {
SuperType.call(this,name);
} var subInstance = new SubType("tongtong",26); subInstance.friends.push('ty'); var subInstance2 = new SubType("fengfeng",27); TestCase("test constructor extends",{
"test subInstance friends property" : function() {
assertEquals("ty",subInstance.friends[2]); //将ty push到数组
},
"test subInstance2 friends length" : function() {
assertEquals(2,subInstance2.friends.length); //subInstance2 friends属性没有改变
}
});

1.4 组合继承(经典伪继承)  -----   推荐模式

将原型链和借用构造函数的技术组合到一块,使用原型链实现对原型属性和方法的继承,通过借用构造函数来实现对实例属性的继承

function SuperType(name) {
this.name = name;
this.friends = ['tong','feng'];
}
SuperType.prototype.sayName = function() {
return this.name;
}; function SubType(name,age) {
//继承属性
SuperType.call(this,name); //调用构造函数
this.age = age;
}
//继承方法
SubType.prototype = new SuperType(); //调用构造函数 SubType.prototype.sayAge = function() {
return this.age;
}; var subInstance = new SubType("tongtong",26); subInstance.friends.push('ty'); var subInstance2 = new SubType("fengfeng",27); TestCase("test extends",{ "test subInstance name property" : function() {
assertEquals("tongtong",subInstance.sayName());
},
"test subInstance2 name property" : function() {
assertEquals("fengfeng",subInstance2.sayName());
},
"test subInstance friends property" : function() {
assertEquals("ty",subInstance.friends[2]); //将ty push到数组
},
"test subInstance2 friends length" : function() {
assertEquals(2,subInstance2.friends.length); //subInstance2 friends属性没有改变
}
});

这种继承的缺点:将会2次调用构造函数,性能一般,解决办法:参见1.7寄生组合式继承

1.5 原型式继承

这种方法没有严格意义上的构造函数,思想是借助原型可以基于已有的对象创建新的对象,同时还不必因此创建自定义类型。

它要求你必须有一个对象可以作为另一个对象的基础。如果有这么一个对象,可以把它传递给object()函数,然后该函数就会返回一个新对象。

function object(o) {
function F() {
} F.prototype = o;
return new F();
} var person = {
name : "tongtong",
friends : [ "feng", "tong" ]
}; var another = object(person);
another.name = "newtong";
another.friends.push('lan'); var other = object(person); TestCase("test prototype extends",{ "test another is extends person name property" : function() {
assertEquals("newtong",another.name);
},
"test person firends property length is 3" : function() {
assertEquals(3,person.friends.length);
},
"test other firends property length is 3" : function() {
assertEquals(3,other.friends.length);
}
});

person 作为另一个对象的基础,我们把它传入到object()中,然后该函数就会返回一个新对象,这个对象将person做为原型。

ECMAScript通过Object.create()方法规范了原型式继承,在传入一个参数的时候,Object.create()与object()方法的行为相同。

var person = {
name : "tongtong",
friends : [ "feng", "tong" ]
};
var another = Object.create(person);
another.name = "lisa";
another.friends.push("lan");
TestCase("test prototype extends",{ "test another is extends person name property" : function() {
assertEquals("lisa",another.name);
},
"test person firends property length is 3" : function() {
assertEquals(3,person.friends.length);
}
});

如果传入2个参数,第二个参数会覆盖同名参数

var person = {
name : "tongtong",
friends : [ "feng", "tong" ]
};
var another = Object.create(person, {
name : {
value : "claire"
}
}); another.friends.push('lalala');
TestCase("test prototype extends",{ "test another is extends person name property" : function() {
assertEquals("claire",another.name);
},
"test person firends property length is 3" : function() {
assertEquals(3,person.friends.length);
}
});

在只想让一个对象与另一个对象保持类似的情况下,又没必要创建构造函数的时候,原型式继承OK.(它和原型模式一样哦,引用类型的值都会被共享)。

1.6寄生式继承

思路:与寄生构造和工厂模式类似,创建一个仅用于封装继承过程的函数,该函数在内部以某种方式来增强对象,最后再像它真的做了所以工作一样返回。

//这是一个函数哦
function createAnother(original) {
// var clone = object(original); //创建一个新对象
var clone = Object.create(original);
clone.sayThis = function() { //增强对象
return original;
};
return clone; //返回对象
}; var person = {
name : "tong",
friends : ['tong','feng']
}; var another = createAnother(person); TestCase("test prototype extends",{
"test another is extends person name property" : function() {
assertEquals("tong",another.sayThis().name);
}
});
;

缺点:使用寄生式继承来为对象添加函数,会因为函数不能复用而降低效率(和构造函数模式相似)

1.7寄生组合式继承  ------  最佳方案

所谓寄生组合式继承,即通过借用构造函数来继承属性,通过原型链的混合形式来继承方法,

思路:不必为了指定子类型的原型而调用超类型的构造函数,我们所需的无非就是超类型原型的一个副本而已。

本质上:使用寄生式继承来继承超类型的原型,然后再将结果指定给子类型的原型

特点:寄生式继承是引用类型最理想的继承方式

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.

缺点:共享属性和方法

寄生式继承:与原型式继承非常相似,基于对象获某些信息创建一个对象,然后增强对象,最后返回对象

优点:解决组合继承多次调用父类的构造函数而导致低效率问题。(可以与组合模式一起使用)

缺点:使用寄生式继承来为对象添加函数,会由于不能做到函数复用而降低效率,与构造函数模式类似。

寄生组合式继承:集寄生式继承和组合继承的优点于一身,是实现基于类型继承的最有效方式。

TDD测试驱动的javascript开发(3) ------ javascript的继承的更多相关文章

  1. 构建简单的Maven工程,使用测试驱动的方式开发项目

    构建简单的Maven工程很简单,这里写这篇随笔的原因是希望自己能记住几个小点. 一.安装Maven 1.下载maven:https://maven.apache.org/download.cgi 2. ...

  2. TDD测试驱动开发

    TDD测试驱动开发 一.概念 TDD故名思意就是用测试的方法驱动开发,简单说就是先写测试代码,再写开发代码.传统的方式是先写代码,再测试,它的开发方式与之正好相反. TDD是极限编程的一个最重要的设计 ...

  3. 我看TDD测试驱动开发

    今天在实验室给大家介绍了一下TDD和Docker,大家对TDD都比较感兴趣,包括老板,也问了一些问题. 还是从头来说TDD吧,TDD作为敏捷开发领域的领头军,充满魅力,同时也充满争议.一切从三大军规说 ...

  4. Scrum敏捷软件开发之技术实践——测试驱动开发TDD

    重复无聊的定义 测试驱动开发,英文全称Test-Driven Development,简称TDD,是一种不同于传统软件开发流程的新型的开发方法.它要求在编写某个功能的代码之前先编写测试代码,然后只编写 ...

  5. 软件工程 - Test-Driven Development (TDD),测试驱动开发

    参考 https://baike.baidu.com/item/%E6%B5%8B%E8%AF%95%E9%A9%B1%E5%8A%A8%E5%BC%80%E5%8F%91/3328831?fr=al ...

  6. 作为JavaScript开发人员,这些必备的VS Code插件你都用过吗?

    本文翻译自:https://www.sitepoint.com/vs-code-extensions-javascript-developers/ 转载请注明出处:葡萄城官网,葡萄城为开发者提供专业的 ...

  7. JavaScript开发工具大全

    译者按: 最全的JavaScript开发工具列表,总有一款适合你! 原文: THE ULTIMATE LIST OF JAVASCRIPT TOOLS 译者: Fundebug 为了保证可读性,本文采 ...

  8. 使用模拟对象(Mock Object)技术进行测试驱动开发

    敏捷开发 敏捷软件开发又称敏捷开发,是一种从上世纪 90 年代开始引起开发人员注意的新型软件开发方法.和传统瀑布式开发方法对比,敏捷开发强调的是在几周或者几个月很短的时间周期,完成相对较小功能,并交付 ...

  9. 为何说 JavaScript 开发很疯狂

    [编者按]本文作者为 Sean Fioritto,主要阐述了 JavaScript 开发为何让人有些无从下手的根本原因.文章系国内 ITOM 管理平台 OneAPM 编译呈现. 网络开发乐趣多多!Ja ...

随机推荐

  1. Ruby学习-第一章

    第一章 字符串,数字,类和对象 为了证明Ruby真的好用,hello world也能写的如此简洁: puts 'hello world' 1.输入/输出 print('Enter your name' ...

  2. zoj 2277 The Gate to Freedom

    N^N = X --->    Nlog10(N) = log10( X ) ---->    X的最高位为 Nlog10(N) 小数部分的第一个非0位 #include<stdio ...

  3. NGUI出现Shader wants normals, but the mesh UIAtlas doesn&#39;t have them

    NGUI出现Shader wants normals, but the mesh UIAtlas doesn't have them,没有网格法线,打开UI Root上 UIPanel组建上的 Nor ...

  4. AIDL使用详解

    一.对AIDL进行说明 AIDL允许定义一个编程的接口来作为客户端和服务端通信的桥梁,AIDL定义了客户端和服务端的编程标准,在Android里边一个进程无法直接访问另一个进程的内存信息,但是要访问的 ...

  5. Aizu 1335 Eequal sum sets

    Let us consider sets of positive integers less than or equal to n. Note that all elements of a set a ...

  6. 解决win7 中source insight没有courier new字节的问题

    解决win7 中source insight没有courier new字节的问题  http://blog.csdn.net/season_hangzhou/article/details/18665 ...

  7. 使用ant的jar任务打jar包

    <?xml version="1.0" encoding="UTF-8"?> <project name="javaTest&quo ...

  8. 高级UIKit-07(AVAudioPlayer)

    [day09-1-AVAudioPlayer]:播放音乐案例 实现多媒体需要准备以下两点: 需要引入一个框架AVFoundation.framework 然后引入#import <AVFound ...

  9. Git 将本次修改追加在上一次修改上面

    Git 将本次修改追加在上一次修改上面 git add . git commit --amend 之后就是进入日志提交页面 确保change-Id那条记录出现在最后一行,如: zh-->en 修 ...

  10. 教你在mac上配置adb环境变量

    1.打开终端,一次输入如下命令 cd ~ touch .bash_profile open -e .bash_profile 2.这时候会在TextEdit中打开一个空白文档,输入下面的语句 a. 输 ...