原型链继承

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. JAVA之旅(十六)——String类,String常用方法,获取,判断,转换,替换,切割,子串,大小写转换,去除空格,比较

    JAVA之旅(十六)--String类,String常用方法,获取,判断,转换,替换,切割,子串,大小写转换,去除空格,比较 过节耽误了几天,我们继续JAVA之旅 一.String概述 String时 ...

  2. C++编译器何时为用户提供默认构造函数

    第一种是类成员中有成员是类对象,并且该成员的类含有默认构造函数,那么C++编译器会帮你给这个类也生成一个默认构造函数,用来调用其成员对象的构造函数,完成该成员的初始化构造.需要强调的是,如果这个成员的 ...

  3. 常用Petri网模拟软件工具简介

    常用Petri网模拟软件工具简介 首先要介绍的的一个非常有名的Petri 网网站--Petri Nets World:       http://www.informatik.uni-hamburg. ...

  4. Android BLE与终端通信(五)——Google API BLE4.0低功耗蓝牙文档解读之案例初探

    Android BLE与终端通信(五)--Google API BLE4.0低功耗蓝牙文档解读之案例初探 算下来很久没有写BLE的博文了,上家的技术都快忘记了,所以赶紧读了一遍Google的API顺便 ...

  5. How to SetUp The Receiving Transaction Manager

    In this Document   Goal   Solution   References APPLIES TO: Oracle Inventory Management - Version: 1 ...

  6. [查阅]MSIL Instruction Set

    Base Instructions   Instruction Description Stack Transition 1 add add two values, returning a new v ...

  7. Android群英传笔记——第十二章:Android5.X 新特性详解,Material Design UI的新体验

    Android群英传笔记--第十二章:Android5.X 新特性详解,Material Design UI的新体验 第十一章为什么不写,因为我很早之前就已经写过了,有需要的可以去看 Android高 ...

  8. LeetCode之“链表”:Reverse Nodes in k-Group

    题目链接 题目要求: Given a linked list, reverse the nodes of a linked list k at a time and return its modifi ...

  9. 避免"Physics Space Locked"错误

    在一些cocos2d中使用物理引擎的代码中,往往会出现如下错误: Aborting due to Chipmunk error: You cannot manually reindex objects ...

  10. 实战项目开发细节:C语言分离一个16进制数取出相应的位1或0

    最近在公司开发一个关于钢琴的PCBA项目,项目大概是这样的,完成各种功能的测试,准备去工厂量产的时候可以通过软件快速甄别硬件是否短路,断路等问题. 其中,甄别好坏的方法是通过比如按键,或者其它的操作然 ...