说在前面:
为了使代码更为简洁方便理解, 本文中的代码均将“非核心实现”部分的代码移出。


一、原型链方式
关于原型链,可点击《深入浅出,JS原型链的工作原理》,本文不再重复叙述。

思路:让子构造函数的原型等于父构造函数的实例


function A() {
}
A.prototype.fn = function (){
console.log("in A");
} function B() {
}
B.prototype = new A(); // 让子构造函数的原型等于父构造函数的实例 var b = new B();
b.fn(); // in A
console.log(b instanceof B); // true
console.log(b instanceof A); // true
console.log(b instanceof Object); // true

缺陷:如果父构造函数中的属性为引用类型,则子构造函数的实例会出现相互影响的情况;


function A() {
this.prop = ['1',"2"];
}
A.prototype.fn = function (){
console.log(this.prop);
} function B() {
}
B.prototype = new A(); var b1 = new B();
var b2 = new B();
b1.fn(); //  ["1", "2"]
b2.fn(); //  ["1", "2"] b1.prop.push('3'); // 子构造函数实例b1修改继承过来的属性
b2.prop.push('4'); // 子构造函数实例b2修改继承过来的属性 b1.fn(); // ["1", "2", "3", "4"] // b2上的修改影响了b1
b2.fn(); // ["1", "2", "3", "4"] // b1上的修改影响了b2

*导致缺陷原因:引用类型,属性变量保存的是地址指针而非实际的值,这个指针指向了一块用来保存实际内容的地址。实例化后,所有实例中变量保存了同一个指针,均指向同一个地址,当任何一个实例通过指针修改地址的内容(并非重新赋予新的指针地址或者修改指针指向)时,其他实例的也会受到影响。


二、借用构造函数方式
为了解决“原型链方式”继承的缺陷,引入的一种“继承”方案。

思路:通过call/apply,在子构造函数中调用父类的构造函数


function A() {
this.prop = ['1',"2"]; this.fn2 = function () {
console.log(this.prop);
}
}
A.prototype.fn = function (){
console.log(this.prop);
} function B() {
A.call(this); // 通过call/apply,在子构造函数中调用父类的构造函数
} var b1 = new B();
var b2 = new B();
b1.fn2(); // ["1", "2"]
b2.fn2(); // ["1", "2"] b1.prop.push('3');
b2.prop.push('4'); b1.fn2(); // ["1", "2", "3"]
b2.fn2(); // ["1", "2", "4"] b1.fn(); // 提示异常:b1.fn is not a function
console.log(b1 instanceof B); // true
console.log(b1 instanceof A); // false
console.log(b1 instanceof Object); // true

缺陷:由于“继承”过程中,A仅充当普通函数被调用,使得父构造函数A原型无法与形成子构造函数B构成原形链关系。因此无法形成继承关系:"b1 instanceof A"结果为false,B的实例b1亦无法调用A原型中的方法。实际意义上,这种不属于继承。


三、组合继承
结合“原型链方式”和“借用构造函数方式”的有点,进行改进的一种继承方式。

思路:原型上的属性和方法通过“原型链方式”继承;父构造函数内的属性和方法通过“借用构造函数方式”继承


function A() {
this.prop = ['1',"2"];
}
A.prototype.fn = function (){
console.log(this.prop);
} function B() {
A.call(this); // 借用构造函数方式
}
B.prototype = new A(); // 原型链方式 var b1 = new B();
var b2 = new B();
b1.fn(); // ["1", "2"]
b2.fn(); // ["1", "2"] b1.prop.push('3');
b2.prop.push('4'); b1.fn(); // ["1", "2", "3"]
b2.fn(); // ["1", "2", "4"]
console.log(b1 instanceof B); // true
console.log(b1 instanceof A); // true
console.log(b1 instanceof Object); // true

缺陷:子构造函数的原型出现一套冗余“父构造函数非原型上的属性和方法”。上述代码在执行“A.call(this);”时候,会给this(即将从B返回给b1赋值的对象)添加一个“prop”属性;在执行“B.prototype = new A();”时,又会通过实例化的形式给B的原型赋值一次“prop”属性。显然,由于实例属性方法的优先级高于原型上的属性方法,绝大多数情况下,原型上的“prop”是不会被访问到的。


四、寄生组合式继承
为了解决“组合继承”中子构造函数的原型链出现冗余的属性和方法,引入的一种继承方式。

思路:在组合继承的基础上,通过Object.create的方式实现原型链方式


function A() {
this.prop = ['1',"2"];
}
A.prototype.fn = function (){
console.log(this.prop);
} function B() {
A.call(this);
}
B.prototype = Object.create(A.prototype); // Object.create的方式实现原型链方式 var b1 = new B();
var b2 = new B();
b1.fn(); // ["1", "2"]
b2.fn(); // ["1", "2"] b1.prop.push('3');
b2.prop.push('4'); b1.fn(); // ["1", "2", "3"]
b2.fn(); // ["1", "2", "4"] console.log(b1 instanceof B); // true
console.log(b1 instanceof A); // true
console.log(b1 instanceof Object); // true

最后补充
1、因为子构造函数的实例自身没有constructor属性,当我们访问实例的constructor属性时,实际是访问原型的constructor属性,该属性应该指向(子)构造函数。到那时上述例子中的代码均会指向父构造函数。为了与ECMAScript规范保持一致,在所有的“原型链继承”后,应当将原型的constructor属性指向子构造函数本身:


B.prototype = ....
--> B.prototype.constructor = B; <--
...

2、Object.create是ECMAScript 5中加入的一个函数,这个函数的功能是:将入参(需为一个对象)作为原型,创建并返回一个新的(只有原型的)的对象。此功能等价于:


function object(o){
function F(){}
F. prototype = o;
return new F();
} // 来源于《JavaScript高级程序设计(第3版)》

来源:https://segmentfault.com/a/1190000017522714

JS中继承方式总结的更多相关文章

  1. js 中继承方式小谈

    题外话 前段时间面试中笔试题有这道题目: 请实现一个继承链,要求如下: 构造函数A():构造函数中有consoleA方法,可以实现console.log("a") 实例对象 a:a ...

  2. js各种继承方式和优缺点的介绍

    js各种继承方式和优缺点的介绍 作者: default 参考网址2 写在前面 本文讲解JavaScript各种继承方式和优缺点. 注意: 跟<JavaScript深入之创建对象>一样,更像 ...

  3. javascript中继承方式及优缺点(三)

    文以<JavaScript高级程序设计>上的内容为骨架,补充了ES6 Class的相关内容,从我认为更容易理解的角度将继承这件事叙述出来,希望大家能有所收获. 1. 继承分类 先来个整体印 ...

  4. js中继承的几种用法总结(apply,call,prototype)

    一,js中对象继承 js中有三种继承方式 1.js原型(prototype)实现继承 <SPAN style="BACKGROUND-COLOR: #ffffff">& ...

  5. js中继承的方法总结(apply,call,prototype)

    一,js中对象继承 js中有三种继承方式 1.js原型(prototype)实现继承 代码如下: <SPAN style="<SPAN style="FONT-SIZE ...

  6. js各种继承方式汇总

    js中的各种继承实现汇总 首先定义一个父类: function Animal(name) { this.name = name || '动物' this.sleep = function () { c ...

  7. 谈谈JavaScript中继承方式

    聊一聊js中的继承 一.简单继承---使用原型赋值的方式继承,将实例化的对象,赋值给子级的原型 父级构造函数 function Parent(param) { this.name = 'parent' ...

  8. js的继承方式分别适合哪些应用场景?

    一.原型链 利用 Person.prototype = new Animal("Human") 实现继承: static式继承.能继承Animal.prototype.不可多重继承 ...

  9. 浅谈js中继承的理解和实现

    一.前言 java.C#等正统面向对象语言都会提供类似extend之类的处理类的继承的方法,而javascript并没有提供专门的方法用于继承,在javascript中使用继承需要一点技巧.js中实例 ...

随机推荐

  1. 解决Ubuntu下gedit中文乱码的情况

    windows下简体中文多用GBK编码 (或GB2312, GB18030), 而linux下多用UTF-8编码. 因此,一般来说在微软平台编辑的中文txt不能在ubuntu下直接打开查看,除非在保存 ...

  2. 卷积 convolution

    这东西大学学过,然后我忘记了,后来就只记得这个名字了. https://zh.wikipedia.org/wiki/%E5%8D%B7%E7%A7%AF http://www.guokr.com/po ...

  3. 2016.7.14 generator基于注解和基于xml自动生成代码的区别

    1.generator配置文件generatorConfig.xml的区别 2.生成代码的区别 注:二者的实体类都一样. (1)基于XML 生成的文件有: 后面省略. 也就是说,基于xml的方式,是要 ...

  4. 多系统启动光盘制作---WIN7+WinXP+老毛桃PE工具箱

    1.工具: ⑴ Windows 7 ISO: ⑵ Windows XP ISO: ⑶ 老毛桃U盘启动盘制作工具V2013 制作得的ISO (含PE.DOS等): ⑷ UltraISO.EasyBoot ...

  5. C语言数据类型的转换

    C语言的类型转换,一个是强制类型进行转换,而在这里要介绍的是自动的数据类型的转换,自动的数据类型转换很多时候是发生在多种数据类型混合使用的时候就会进行类型的转换,这样就会带来不能控制的结果,所以必须进 ...

  6. 解决Linux中文环境下astro和Calibre不能输入的问题

    例如我的opensuse在中文环境下不能在astro中输入指令,Calibre的grid spacing设置框不能输入,经过摸索,找到以下两种解决方法: 1.    将系统环境变成英文,在.bashr ...

  7. 开发ActiveX控件调用另一个ActiveX系列0——身份证识别仪驱动的问题

    程序员要从0下表开始,这篇是介绍这个系列的背景的,没有兴趣的人可以直接跳过. 为什么要开发ActiveX控件 由于工作需要,我们开发了一个网站,使用了一款身份证识别仪的网页ActiveX(OCX)插件 ...

  8. 利用GROUP_CONCAT函数把相同信息的合并到同一个字段中

    SELECT a.*,GROUP_CONCAT(b.pri_name) FROM sh_role a LEFT JOIN sh_privilege b ON FIND_IN_SET(b.id,a.pr ...

  9. 02 redis通用命令操作

    set hi hello 设置值 get hi 获取值 keys * 查询出所有的key memcached 不能查询出所有的key keys *h 模糊查找key keys h[ie] 模糊查找 k ...

  10. ios --转载获ipa 的图片资源

      突然想起当初刚学习iOS的时候,就经常通过抓包和提取素材的方式来模仿App,今天就教大家如何一步步提取App的素材! 大家是否有过想要获取别人的素材的想法?看到某些App的资源很不错,很想导出来用 ...