深入理解Javascript的继承和原型链_第1页
深入理解Javascript的继承和原型链_第2页
深入理解Javascript的继承和原型链_第3页
深入理解Javascript的继承和原型链_第4页
深入理解Javascript的继承和原型链_第5页
已阅读5页,还剩2页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

1、深入理解Javascript的继承和原型链 具体代码如下: function SuperClass() = "women" SuperCtotype.sayWhat = function() return + ":im a girl!" function SubClass() this.subname = "your sister" SubCtotype = new SuperClass(); SubCtotype.subSayWhat = fun

2、ction() return this.subname + ":im a beautiful girl" var sub = new SubClass(); console.log(sub.sayWhat();/women:im a girl!使用原型链实现继承通过上面的代码中可以看出SubClass继承了SuperClass的属性和方法,这个继承的实现是通过将SuperClass的实例赋值给SubClass的原型对象,这样SubClass的原型对象就被SuperClass的一个实例覆盖掉了,拥有了它的全部属性和方法,同时还拥有一个指向SuperClass原型对象的指针。在

3、使用原型链实现继承时有一些需要我们注意的地方:注意继承后constructor的变化。此处sub的constructor指向的是SuperClass,因为SubClass的原型指向了SuperClass的原型。在了解原型链时,不要忽略掉在末端还有默认的Object对象,这也是我们能在所有对象中使用toString等对象内置方法的原因。通过原型链实现继承时,不能使用字面量定义原型方法,因为这样会重写原型对象(在上一篇文章中也介绍过):function SuperClass() = "women" SuperCtotype.sayWhat =

4、 function() return + ":im a girl!" function SubClass() this.subname = "your sister" SubCtotype = new SuperClass(); SubCtotype = /此处原型对象被覆盖,因为无法继承SuperClass属性和方法 subSayWhat:function() return this.subname + ":im a beautiful girl" var sub = new Su

5、bClass(); console.log(sub.sayWhat();/TypeError: undefined is not a function实例共享的问题。在前面讲解原型和构造函数时,我们曾经介绍过包含引用类型属性的原型会被所有的实例共享,同样,我们继承而来的原型中也会共享“父类”原型中引用类型的属性,当我们通过原型继承修改了“父类”的引用类型属性后,其他所有继承自该原型的实例都会受到影响,这不仅浪费了资源,也是我们不愿看到的现象:function SuperClass() = "women" this.bra = "a",

6、"b" function SubClass() this.subname = "your sister" SubCtotype = new SuperClass(); var sub1 = new SubClass(); = "man" sub1.bra.push("c"); console.log();/man console.log(sub1.bra);/"a","b","c" var sub2 =

7、 new SubClass(); console.log();/woman console.log(sub2.bra);/"a","b","c"注意:此处在数组中添加一个元素,所有继承自SuperClass的实例都会受到影响,但是如果修改name属性则不会影响到其他的实例,这是因为数组为引用类型,而name为基本类型。如何解决实例共享的问题呢?我们接着往下看.经典继承(constructor stealing)正如我们介绍过很少单独使用原型定义对象一样,在实际开发中我们也很少单独使用原型链,为了解决引用类型的共享问题

8、,javascript开发者们引入了经典继承的模式(也有人称为借用构造函数继承),它的实现很简单就是在子类型构造函数中调用超类型的构造函数。我们需要借助javascript提供的call()或者apply()函数,我们看下示例:function SuperClass() = "women" this.bra = "a", "b"function SubClass() this.subname = "your sister" /将SuperClass的作用域赋予当前构造函数,实现继承 SuperC

9、lass.call(this);var sub1 = new SubClass();sub1.bra.push("c");console.log(sub1.bra);/"a","b","c"var sub2 = new SubClass();console.log(sub2.bra);/"a","b"SuperClass.call(this);这一句话的意思是在SubClass的实例(上下文)环境中调用了SuperClass构造函数的初始化工作,这样每一个实例就会有自己的一份

10、bra属性的副本了,互不产生影响了。但是,这样的实现方式仍不是完美的,既然引入了构造函数,那么同样我们也面临着上篇中讲到的构造函数存在的问题:如果在构造函数中有方法的定义,那么对于没一个实例都存在一份单独的Function引用,我们的目的其实是想共用这个方法,而且我们在超类型原型中定义的方法,在子类型实例中是无法调用到的:function SuperClass() = "women" this.bra = "a", "b" SuperCtotype.sayWhat = function() cons

11、ole.log("hello"); function SubClass() this.subname = "your sister" SuperClass.call(this); var sub1 = new SubClass(); console.log(sub1.sayWhat();/TypeError: undefined is not a function如果你看过上篇文章关于原型对象和构造函数的,想必你已经知道解决这个问题的答案了,那就是沿用上篇的套路,使用“组合拳”!组合式继承组合式继承就是结合原型链和构造函数的优势,发出各自特长,组合起来实

12、现继承的一种方式,简单来说就是使用原型链继承属性和方法,使用借用构造函数来实现实例属性的继承,这样既解决了实例属性共享的问题,也让超类型的属性和方法得到继承:function SuperClass() = "women" this.bra = "a", "b" SuperCtotype.sayWhat = function() console.log("hello"); function SubClass() this.subname = "your sister&qu

13、ot; SuperClass.call(this); /第二次调用SuperClass SubCtotype = new SuperClass(); /第一次调用SuperClass var sub1 = new SubClass(); console.log(sub1.sayWhat();/hello组合继承的方式也是实际开发中我们最常用的实现继承的方式,到此已经可以满足你实际开发的需求了,但是人对完美的追求是无止境的,那么,必然会有人对这个模式“吹毛求疵”了:你这个模式调用了两次超类型的构造函数耶!两次耶。你造吗,这放大一百倍是多大的性能损失吗?最有力的反驳莫过于拿出解决方

14、案,好在开发者找到了解决这个问题的最优方案:寄生组合式继承在介绍这个继承方式前,我们先了解下寄生构造函数的概念,寄生构造函数类似于前面提到的工厂模式,它的思想是定义一个公共函数,这个函数专门用来处理对象的创建,创建完成后返回这个对象,这个函数很像构造函数,但构造函数是没有返回值的:function Gf(name,bra) var obj = new Object(); = name; obj.bra = bra; obj.sayWhat = function() console.log(); return obj;var gf1 = new Gf(&quo

15、t;bingbing","c+");console.log(gf1.sayWhat();/bingbing寄生式继承的实现和寄生式构造函数类似,创建一个不依赖于具体类型的“工厂”函数,专门来处理对象的继承过程,然后返回继承后的对象实例,幸运的是这个不需要我们自己实现,道哥(道格拉斯)早已为我们提供了一种实现方式:function object(obj) function F() F.prototype = obj; return new F();var superClass = name:"bingbing", bra:"c+&quo

16、t;var subClass = object(superClass);console.log(subC);/bingbing在公共函数中提供了一个简单的构造函数,然后将传进来对象的实例赋予构造函数的原型对象,最后返回该构造函数的实例,很简单,但疗效很好,不是吗?这个方式被后人称为“原型式继承”,而寄生式继承正是在原型式基础上,通过增强对象的自定义属性实现的:function buildObj(obj) var o = object(obj); o.sayWhat = function() console.log("hello"); return o;va

17、r superClass = name:"bingbing", bra:"c+"var gf = buildObj(superClass);gf.sayWhat();/hello寄生式继承方式同样面临着原型中函数复用的问题,于是,人们又开始拼起了积木,诞生了寄生组合式继承,目的是解决在指定子类型原型时调用父类型构造函数的问题,同时,达到函数的最大化复用。基于以上基础实现方式如下:/参数为两个构造函数function inheritObj(sub,sup) /实现实例继承,获取超类型的一个副本 var proto = object(totyp

18、e); /重新指定proto实例的constructor属性 proto.constructor = sub; /将创建的对象赋值给子类型的原型 totype = proto;function SuperClass() = "women" this.bra = "a", "b"SuperCtotype.sayWhat = function() console.log("hello");function SubClass() this.subname = "your sister" SuperClass.call(this);inheritObj(SubClass,SuperClass);var sub1

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论