组合继承

  组合继承(combination inheritance),有时候也叫做伪经典继承,指的是将原型链和借用构造函数的 技术组合到一块,从而发挥二者之长的一种继承模式。其背后的思路是使用原型链实现对原型属性和方 法的继承,而通过借用构造函数来实现对实例属性的继承。这样,既通过在原型上定义方法实现了函数 复用,又能够保证每个实例都有它自己的属性。下面来看一个例子:

  

   <script>
function Person(name, age, friends) {
this.name = name;
this.age = age;
this.friends = ["Carol", "Seylia"];
if (typeof this.ShowPerson != "function") {
this.ShowPerson = function () {
var message = "name:" + this.name + "\n\rage:" + this.age + "\r\nfriends:";
this.friends.forEach(function (value, index, arr) {
message += " " + value;
});
alert(message);
}
}
} function Student(name, age, job) {
Person.call(this, name, age);
this.job = job;
} //这里继承了
Student.prototype = new Person();
Student.prototype.constructor = Student; var s1 = new Student("gck", , "Sf");
s1.friends.push("Core");
s1.ShowPerson(); var s2 = new Student("Carol", , "Sf");
s2.ShowPerson();
</script>

组合继承避免了原型链和借用构造函数的缺陷,融合了它们的优点,成为 JavaScript中常用的继 承模式。而且,instanceof 和 isPrototypeOf()也能够用于识别基于组合继承创建的对象.

  注:不准备说原型链和借用构造,因为这两个缺陷太大,不过有想了解的可以查一下。

寄生组合式继承

面说过,组合继承是 JavaScript 常用的继承模式;不过,它也有自己的不足。组合继承大的 问题就是无论什么情况下,都会调用两次超类型构造函数:一次是在创建子类型原型的时候,另一次是 在子类型构造函数内部。没错,子类型终会包含超类型对象的全部实例属性,但我们不得不在调用子 类型构造函数时重写这些属性,所以我用寄生组合继承:

  所谓寄生组合式继承,即通过借用构造函数来继承属性,通过原型链的混成形式来继承方法。其背 后的基本思路是:不必为了指定子类型的原型而调用超类型的构造函数,我们所需要的无非就是超类型 原型的一个副本而已。本质上,就是使用寄生式继承来继承超类型的原型,然后再将结果指定给子类型 的原型。寄生组合式继承的基本模式如下所示。

<script>
//如果Object.create浏览器不支持,则用这个方法代替
function object(o) {
function F() { }
F.prototype = o;
return new F();
} function inheritPrototype(subType, superType) { //创建对象
var prototype = Object.create(superType.prototype);//这个方法IE9+才支持
//var prototype = object(superType.prototype); //增强对象
prototype.constructor = subType; //指定对象
subType.prototype = prototype; } function Person(name, age, friends) {
this.name = name;
this.age = age;
this.friends = ["Carol", "Seylia"];
if (typeof this.ShowPerson != "function") {
this.ShowPerson = function () {
var message = "name:" + this.name + "\n\rage:" + this.age + "\r\nfriends:";
this.friends.forEach(function (value, index, arr) {
message += " " + value;
});
alert(message);
}
}
} function Student(name, age, job) {
Person.call(this, name, age);
this.job = job;
} //这里继承了
inheritPrototype(Student, Person); var s1 = new Student("gck", 26, "Sf");
s1.friends.push("Core");
s1.ShowPerson(); var s2 = new Student("Carol", 28, "Sf");
s2.ShowPerson(); </script>

这个例子的高效率体现在它只调用了一次Person构造函数,并且因此避免了在 Student. prototype 上面创建不必要的、多余的属性。与此同时,原型链还能保持不变;因此,还能够正常使用 instanceof 和 isPrototypeOf()。开发人员普遍认为寄生组合式继承是引用类型理想的继承范式。
YUI

最新文章

  1. MVC项目中使用js 设置Checkbox的选中事件
  2. PHP设计的超强大的文件上传类(单文件上传)
  3. Focus
  4. js树形控件
  5. javaSwing文本框组件
  6. osmdroid启程
  7. (1)opengl-nehe 4种框架
  8. OSG显示文字——自定义显示文字函数
  9. Android:刚6瓶啤酒4两56度白酒下肚,竟然20分钟做了一手机版站点 !
  10. echarts学习总结(一):图表溢出窗口,图表数据窗口显示不全
  11. 201521123108《Java程序设计》第1周学习总结
  12. 用Arduino制作一个二维码显示器
  13. response.redirect 与location.href 的区别
  14. c++ 第一次实验
  15. win32: 查询滚动条相关信息的注意事项
  16. session操作类
  17. C++解析四-友员函数、内联函数、静态成员
  18. [翻译] TSMessages
  19. Hadoop学习之路(二十六)MapReduce的API使用(三)
  20. loadrunner socket协议问题归纳(6)

热门文章

  1. [转]Android 超高仿微信图片选择器 图片该这么加载
  2. BSTestRunner——一个丑在路上的python unnitest HTML报告生成Runner
  3. windows7系统下如何安装windows xp系统(无法识别硬盘,删除隐藏分区)
  4. php 生成 Json
  5. vim中多标签和多窗口的使用
  6. pecl install imagick
  7. oracle 数据库时间类型为字符串 时间范围大小查询
  8. Git回滚远程版本
  9. [转]android开发之字节顺序
  10. preg_match_all正则表达式的基本使用