JavaScript继承
最佳的继承范式
寄生组合继承
我们来看一下它的实现方式:
function Object(o){
var TempObject = function(){};
TempObject.prototype = o;
return new TempObject();
}
function inheritPrototype(subType,superType){
subType.prototype = Object(superType.prototype);
subType.prototype.constructor = subType;
}
function Person(name,age){
this.name = name;
this.age = age;
this.hobby = ["football","swimming"];
}
Person.prototype.sayName = function(){
console.log(this.name);
}
function Teacher(name,age,subject){
Person.call(this,name,age);
this.subject = subject;
}
inheritPrototype(Teacher,Person);
var t1 = new Teacher("Tom","32","English");
var t2 = new Teacher("Jane","28","Math");
t1.sayName();
t2.sayName();
t1.hobby.push("cooking");
t2.hobby.push("eat");
console.log(t1.hobby);
console.log(t2.hobby);
console.log(Teacher.prototype.hobby);
运行结果:
Tom
Jane
Array [ "football", "swimming", "cooking" ]
Array [ "football", "swimming", "eat" ]
undefined
这里面结合了寄生式继承(来源于原型式继承)和组合继承,原型式继承可以以一个对象为基础,对基础对象进行浅复制,然后赋给子类。寄生式实现对原型式继承的封装,使之拓展更多的属性和方法。而组合继承解决了原型链对于原型对象全面复制父类构造函数的属性的问题(特别是引用类型,从构造函数的属性变成原型对象的属性,就共享了),同时解决了借用构造函数只能把属性和方法写入构造函数中的问题。组合继承的问题在于原型对象中有多余的属性(因为使用了原型链继承,构造函数的属性都继承在了原型中了,这个问题由原型式继承解决)。
将上述的这些继承方式结合起来,扬长避短,就有了寄生组合继承,解决了上述所描述的问题,所以它是最佳的继承实现方式。
这里简单分析一下:
function Object(o){
var TempObject = function(){};
TempObject.prototype = o;
return new TempObject();
}
function inheritPrototype(subType,superType){
subType.prototype = Object(superType.prototype);
subType.prototype.constructor = subType;
}
第一个函数Object(o)传入一个对象,在函数里面创立临时构造函数TempObject,并把对象o赋值给临时构造函数的原型TempObject.prototype,最后返回它的实例,结果是返回的实例中有一个[[prototype]]的指针指向它自身的原型对象,而且这个原型对象里面的属性就是o的属性。为了描述方便,这里我们把这个返回的实例称之为Tab(随便起的名字),用于下面的描述。
在第二个函数中,subType.prototype = Object(superType.prototype);
表示把superType的原型通过Object拷贝一份后赋值给Tab的原型对象,而subType的原型subType.prototype就等于这个实例。即subType.prototype有个[[prototype]]指针指向另外一个原型对象(即被继承的原型对象)。当然,此时subType.prototype在继承后也可以有自己方法。这个方法可以写在inheritPrototype() 里面(实现封装),也可以在这个函数外面增加自己的方法。
最后,注意一下运行结果的最后一个结果:undefined,因为通过寄生式继承,在原型对象中,只会继承父类的原型对象,自然在里面就找不到构造函数的属性了。
请看下面的例子(组合继承),这里就存在上述的弊端,即原型对象包含了父类构造函数的属性,它之所以能够不共享这些属性,仅仅只是因为在创建实例的时候创建了同样的属性覆盖了这些共享属性罢了。
// 组合继承
// 父类Person
function Person(name,age){
this.name = name;
this.age = age;
this.hobby = ["football","swimming"];
}
Person.prototype = {
constructor:Person,
sayName:function(){
console.log("Hi,"+this.name);
}
}
// 子类Teacher
function Teacher(name,age,subject){
Person.call(this,name,age);
this.subject = subject;
}
Teacher.prototype = new Person();
var t1 = new Teacher("Tom","32","English");
var t2 = new Teacher("Jane","28","math");
t1.sayName();
console.log(t1.age);
t1.hobby.push("cooking");
console.log(t1.hobby);
console.log("----------------------");
t2.sayName();
console.log(t2.age);
t2.hobby.push("eat");
console.log(t2.hobby);
console.log(Teacher.prototype.hobby);
运行结果:
Hi,Tom
32
Array [ "football", "swimming", "cooking" ]
----------------------
Hi,Jane
28
Array [ "football", "swimming", "eat" ]
Array [ "football", "swimming" ]
注意到最后一个Array [ "football", "swimming" ],可见组合继承中的原型对象存在多余的属性(来自构造函数)。
参考
- 《JavaScript高级程序设计》
JavaScript继承的更多相关文章
- javascript继承的三种模式
javascript继承一般有三种模式:组合继承,原型式继承和寄生式继承: 1组合继承:javascript最为广泛的继承方式通过原型链实现对原型属性和方法的继承,通过构造函数实现对实例属性的继承,同 ...
- javascript继承机制的设计思想(ryf)
我一直很难理解Javascript语言的继承机制. 它没有"子类"和"父类"的概念,也没有"类"(class)和"实例" ...
- 【读书笔记】javascript 继承
在JavaScript中继承不像C#那么直接,C#中子类继承父类之后马上获得了父类的属性和方法,但JavaScript需要分步进行. 让Brid 继承 Animal,并扩展自己fly的方法. func ...
- 图解JavaScript 继承
JavaScript作为一个面向对象语言,可以实现继承是必不可少的,但是由于本身并没有类的概念(不知道这样说是否严谨,但在js中一切都类皆是对象模拟)所以在JavaScript中的继承也区别于其他的面 ...
- JavaScript强化教程——Cocos2d-JS中JavaScript继承
javaScript语言本身没有提供类,没有其它语言的类继承机制,它的继承是通过对象的原型实现的,但这不能满足Cocos2d-JS引擎的要求.由于Cocos2d-JS引擎是从Cocos2d-x演变而来 ...
- [原创]JavaScript继承详解
原文链接:http://www.cnblogs.com/sanshi/archive/2009/07/08/1519036.html 面向对象与基于对象 几乎每个开发人员都有面向对象语言(比如C++. ...
- javascript继承(六)—实现多继承
在上一篇javascript继承—prototype最优两种继承(空函数和循环拷贝)(3) ,介绍了js较完美继承的两种实现方案,那么下面来探讨一下js里是否有多继承,如何实现多继承.在这里可以看看j ...
- javascript继承(五)—prototype最优两种继承(空函数和循环拷贝)
一.利用空函数实现继承 参考了文章javascript继承—prototype属性介绍(2) 中叶小钗的评论,对这篇文章中的方案二利用一个空函数进行修改,可以解决创建子类对象时,父类实例化的过程中特权 ...
- javascript继承(四)—prototype属性介绍
js里每一个function都有一个prototype属性,而每一个实例都有constructor属性,并且每一个function的prototype都有一个constructor属性,这个属性会指向 ...
- 【JavaScript】重温Javascript继承机制
上段时间,团队内部有过好几次给力的分享,这里对西风师傅分享的继承机制稍作整理一下,适当加了些口语化的描述,留作备案. 一.讲个故事吧 澄清在先,Java和Javascript是雷锋和雷峰塔的关系.Ja ...
随机推荐
- iOS网络相关知识总结
iOS网络相关知识总结 1.关于请求NSURLRequest? 我们经常讲的GET/POST/PUT等请求是指我们要向服务器发出的NSMutableURLRequest的类型; 我们可以设置Reque ...
- h5 与app交互
http://www.jianshu.com/p/7151987f012d JSContext *context = [self.webView valueForKeyPath:@"docu ...
- ubuntu14.04下安装python3.4.2
1. python安装包的下载地址:https://www.python.org/downloads/ 我的python安装包下载地址:https://www.python.org/ftp/pytho ...
- OEIS A140358
以前也许做过? 有点方 最小整数1到k 加减得到 n 1+-2+-3+-...+-k = n 求最小k #include <cstdio> #include <algorithm&g ...
- 收集一些关于OI/ACM的奇怪的东西……
一.代码: 1.求逆元(原理貌似就是拓展欧几里得,要求MOD是素数): int inv(int a) { if(a == 1) return 1; return ((MOD - MOD / a) * ...
- java方法与构造
public class StaticClass { static int i = 50; int y; // 构造只能有访问修饰符public.protected.private 修饰 (访问修饰符 ...
- c# 用户名 密码 访问 局域网共享
#region Ping 返回true则代表可以连接成功 public bool Ping(string remoteHost) { bool Flag = false; Process proc = ...
- zw.delphi不同版本程序运行速度测试
{ zw.delphi不同版本程序运行速度测试 delphi无论是开发,编译,还是运行,速度方面向来不差,笔者很少进行这种微粒度的优化,调试. 最近,因为项目需要,发现:同一个函数模块,差不多同样的代 ...
- js 小数相加
<!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xhtml"> <head runat=&qu ...
- Oracle数据访问组件ODAC的安装方法
Oracle数据访问组件ODAC(Oracle Data Access Components)顾名思义就是用来访问Oracle数据库的小程序.我们可以编程调用这些组件来实现在没有安装Oracle数据库 ...