原型链继承

function SuperType(){
this.property = true;
} SuperType.prototype.getSuperValue = function(){
return this.property;
}; function SubType(){
this.subproperty = false;
} // 继承了 SuperType
SubType.prototype = new SuperType(); var subType = new SubType();
console.log(subType.getSuperValue()); // 继承了 SuperType 的 getSuperValue 方法,打印 true

缺点

  1. 如果 SuperType 存在一个引用类型的属性,而 SubType 的原型对象变为 SuperType 的一个实例,这样每个 SubType 的实例都会共用这个引用类型的属性,不同的 SubType 实例对该属性的操作都将会在其它 SubType 实例中体现出来,这跟每个实例拥有自己的属性是违背的。
function SuperType(){
this.colors = ["red", "blue", "green"];
} function SubType(){
} // 继承了 SuperType
SubType.prototype = new SuperType(); var subType1 = new SubType();
subType1.colors.push("black"); // subType1 修改 colors
console.log(subType1.colors); // "red,blue,green,black" var subType2 = new SubType();
console.log(subType2.colors); // "red,blue,green,black",subType2 的 colors 值为 subType1 修改之后的值
  1. 在创建子类型(SubType)的实例时,不能向超类型(SuperType)的构造函数中传递参数。实际上,应该说是没有办法在不影响所有对象实例(原因如缺点1)的情况下,给超类型的构造函数传递参数。

借用构造函数继承

function SuperType(){
this.colors = ["red", "blue", "green"];
} function SubType(){
// 继承了 SuperType
SuperType.call(this); // 通过 apply() 或 call() 执行 SuperType 的构造函数
} var instance1 = new SubType();
instance1.colors.push("black");
alert(instance1.colors); // "red,blue,green,black" var instance2 = new SubType();
alert(instance2.colors); // "red,blue,green"

该方法解决了 原型链继承 的引用型属性共享的问题。

还有可以在子类型构造函数中向超类型构造函数传递参数,如下例子:

function SuperType(name){
this.name = name;
} function SubType(){
// 继承 SuperType,并传递参数
SuperType.call(this, "Nicholas"); // 实例属性
this.age = 29;
} var subType = new SubType();
console.log(subType.name); //"Nicholas";
console.log(subType.age); //29

缺点

  1. 方法都在构造函数中定义,因此函数复用就无从谈起了。而且,在超类型的原型中定义的方法,对子类型而言也是不可见的,结果所有类型都只能使用构造函数模式。

原型链/构造函数组合继承

function SuperType(name){
this.name = name;
this.colors = ["red", "blue", "green"];
} SuperType.prototype.sayName = function(){
console.log(this.name);
}; function SubType(name, age){
// 执行 SuperType 的构造函数,继承 SuperType 的属性
SuperType.call(this, name); this.age = age;
} // 将 SuperType 的实例赋给 SubType 的原型对象,这样 SubType 可继承 SuperType 原型中的方法
SubType.prototype = new SuperType(); SubType.prototype.sayAge = function(){
console.log(this.age);
}; var subType1 = new SubType("Nicholas", 29);
subType1.colors.push("black");
console.log(subType1.colors); // "red,blue,green,black"
subType1.sayName(); // "Nicholas";
subType1.sayAge(); // 29 var subType2 = new SubType("Greg", 27);
console.log(subType2.colors); // "red,blue,green"
subType2.sayName(); // "Greg";
subType2.sayAge(); // 27

缺点

  1. 无论什么情况,都会调用两次超类型构造函数,一次是在创建子类型原型的时候,一次是在执行子类型构造函数的时候。这样一个 SubType 实例会包含两组 SuperType 的属性,一组在 SubType 实例上,一组在 SubType 原型中。

原型式继承

该方法基于已有的对象创建新对象,同时还不必因此创建自定义类型。

var person = {
name: "Nicholas",
friends: ["Shelby", "Court", "Van"]
}; var anotherPerson = Object.create(person); // 使用 person 作为新对象(anotherPerson)的原型
anotherPerson.name = "Greg";
anotherPerson.friends.push("Rob"); var yetAnotherPerson = Object.create(person);
yetAnotherPerson.name = "Linda";
yetAnotherPerson.friends.push("Barbie"); console.log(person.friends); // "Shelby,Court,Van,Rob,Barbie"

寄生式继承

该方法创建一个仅用于封装继承过程的函数,该函数在内部以某种方式来增强对象,最后再像真地是它做了所有工作一样返回对象。

function creatAnother(original) {
var clone = Object.create(original); // 创建一个新对象
clone.sayHi = function() { // 以某种方式来增强这个对象
console.log("hi");
}
return clone; // 返回该对象
} // 使用示例
var person = {
name: "Nicholas",
friends: ["Shelby", "Court", "Van"]
}; var anotherPerson = creatAnother(person);
anotherPerson.sayHi(); // "hi"

寄生/原型链/构造函数组合继承

该方法解决原型链/构造函数组合继承调用两次超类型构造函数的问题。

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

function object(o){
function F(){}
F.prototype = o;
return new F();
} function inheritPrototype(subType, superType){
var prototype = object(superType.prototype); // 根据超类型原型创建新对象
prototype.constructor = subType; // 将新对象的 constructor 设置为子类型
subType.prototype = prototype; // 将新对象赋给子类型的原型
} function SuperType(name){
this.name = name;
this.colors = ["red", "blue", "green"];
} SuperType.prototype.sayName = function(){
console.log(this.name);
}; function SubType(name, age){
SuperType.call(this, name); this.age = age;
} inheritPrototype(SubType, SuperType); SubType.prototype.sayAge = function(){
console.log(this.age);
}; var subType1 = new SubType("Nicholas", 29);
subType1.colors.push("black");
console.log(subType1.colors); // "red,blue,green,black"
subType1.sayName(); // "Nicholas";
subType1.sayAge(); // 29 var subType2 = new SubType("Greg", 27);
alert(subType2.colors); // "red,blue,green"
subType2.sayName(); // "Greg";
subType2.sayAge(); // 27

参考资料:《JavaScript高级程序设计(第3版)》第6.3节 继承

JavaScript之对象继承的更多相关文章

  1. JavaScript 的对象继承方式,有几种写法?

    JavaScript 的对象继承方式,有几种写法? 一.对象冒充 其原理如下:构造函数使用 this 关键字给所有属性和方法赋值(即采用类声明的构造函数方式).因为构造函数只是一个函数,所以可使 Pa ...

  2. 架构师JavaScript 的对象继承方式,有几种程序写法?

    架构师JavaScript 的对象继承方式,有几种程序写法?   一.对象冒充 其原理如下:构造函数使用 this 关键字给所有属性和方法赋值(即采用类声明的构造函数方式).因为构造函数只是一个函数, ...

  3. 浏览器环境下的javascript DOM对象继承模型

    这张图是我直接在现代浏览器中通过prototype原型溯源绘制的一张浏览器宿主环境下的javascript DOM对象模型,对于有效学习和使用javascript DOM编程起到高屋建瓴的指导作用, ...

  4. 关于JavaScript中对象的继承实现的学习总结

    一.原型链 JavaScript 中原型链是实现继承的主要方法.其主要的思想是利用原型让一个引用类型继承另一个引用类型的属性和方法.实现原型链有一种基本模式,其代码如下. function Super ...

  5. JS中对象继承方式

    JS对象继承方式 摘自<JavaScript的对象继承方式,有几种写法>,作者:peakedness 链接:https://my.oschina.net/u/3970421/blog/28 ...

  6. JavaScript创建对象及对象继承

    面向对象的语言有一个标志,那就是他们都有类的概念,而通过类可以创建任意多个具有相同属性和方法的对象.但是在ECMAScript中没有类的概念,因此它的对象也与基于类的对象有所不同.实际上,JavaSc ...

  7. javascript对象继承的实现

    现在有两个对象,需要实现Chinese类型对象对Person类型对象的继承. 这里分两部分,属性和方法. 属性可以直接用构造函数的方法实现继承,而方法则要通过原型链来实现继承. 先解释什么是原型链,每 ...

  8. Javascript实现对象的继承

    在Java和C#中,你可以简单的理解class是一个模子,对象就是被这个模子压出来的一批一批月饼.压个啥样,就得是个啥样,不能随便动,动一动就坏了.而在Javascript中,没有模子,月饼被换成了面 ...

  9. 【JavaScript】类继承(对象冒充)和原型继承__深入理解原型和原型链

    JavaScript里的继承方式在很多书上分了很多类型和实现方式,大体上就是两种:类继承(对象冒充)和原型继承. 类继承(对象冒充):在函数内部定义自身的属性的方法,子类继承时,用call或apply ...

随机推荐

  1. Chapter 1 Securing Your Server and Network(12):保护链接服务器

    原文出处:http://blog.csdn.net/dba_huangzj/article/details/38438363,专题目录:http://blog.csdn.net/dba_huangzj ...

  2. STL:字符串用法详解

    字符串是程序设计中最复杂的变成内容之一.STL string类提供了强大的功能,使得许多繁琐的编程内容用简单的语句就可完成.string字符串类减少了C语言编程中三种最常见且最具破坏性的错误:超越数组 ...

  3. Guava 教程(3):Java 的函数式编程,通过 Google Collections 过滤和调用

    原文出处: oschina 在本系列博客的第一.二部分,我介绍了非常优秀的Google Collections和Guava包.本篇博客中我们来看看如何使用Google Collections来做到过滤 ...

  4. Django介绍、安装配置、基本使用、Django用户注册例子

    Django介绍     Django 是由 Python 开发的一个免费的开源网站框架,可以用于快速搭建高性能,优雅的网站     DjangoMTV的思想         没有controller ...

  5. Linux 操作之基础命令

    1.罗列出文件和文件夹 –ls ls 是帮助我们罗列出当前目录下的所有的文件和文件夹,当然了,还可以加上许多选项,最为重要的是所加的参数可以进行组合,起到让人意想不到的效果,下面就是常用的一些ls的及 ...

  6. Android官方技术文档翻译——Gradle 插件用户指南(6)

    没想到翻译这篇<Gradle 插件用户指南>拖了差不多一个月,还跨年了.不过还好,在2号时终于一口气把剩下的给翻译完了(其实那天剩下的也就不到一章). 今天先发一下第六章,明天再发第七章. ...

  7. MPLSVPN 命令集

      载请标明出处:http://blog.csdn.net/sk719887916,作者:skay 读懂下面配置命令需要有一定的TCP/IP,路由协议基础,现在直接上关键VPN命令. router ...

  8. 01_MUI之Boilerplate中:HTML5示例,动态组件,自定义字体示例,自定义字体示例,图标字体示例

     1安装HBuilder5.0.0,安装后的界面截图如下: 2 按照https://www.muicss.com/docs/v1/css-js/boilerplate-html中的说明,创建上图的 ...

  9. 《java入门第一季》之面向对象(构造方法)

    /* 构造方法: 给对象的数据进行初始化 格式: A:方法名与类名相同 B:没有返回值类型,连void都没有 C:没有具体的返回值 */ class Student { private String ...

  10. Java-ServletContextEvent-ServletContextAttributeEvent

    //这是一个事件类用来通知一个web 应用的servlet 上下文的改变 public class ServletContextEvent extends java.util.EventObject ...