继承是我们在实现面向对象编程的时候很重要的一个手段。虽然我们讲不能过度继承,多利用组合代替继承,但是继承总是免不了的。这里要讨论的就是Javascript中的继承机制。
Javascript中实际上是没有继承的概念的,但是我们可以通过一些手段来模仿实现它。这种继承实际上把一个对象复制到另外一个对象内部。你需要注意的是所有的本地类和宿主类是不能作为基类被继承的,主要是为了安全方面的考虑。
Javascript中的继承大约有三类:1.对象冒充;2.原型继承;3.二者的混合。
一、对象冒充
其实对象冒充是跟this关键字紧密联系在一起的(所以说充分理解Javascript中的this关键字是多么的重要:P)。构造函数使用this来给属性和方法赋值,而构造函数也可以看作为一个普通的函数,所以我们就可以使我们的基类的构造函数成为子类的构造函数,然后在子类的内部调用这个函数,那么子类就会得到父类的属性和方法。
原理很简单,那我们怎么实现呢?下面就以代码示例,实际的操作一下。
对象冒充实现方法一,我们最常用的新建对象的方法:
复制代码 代码如下:
var classA = function(name){
this.name = name;
this.alertName = function(){
alert(this.name);
}
}
var classB = function(name,age){
this.myConstructor = classA;
this.myConstructor(name);
delete this.myConstructor;
this.age = age;
this.alertAge = function(){
alert(this.age);
}
}
为了验证以上的方法是否正确,你可以亲自测试下,我将测试用的代码写在下面:
复制代码 代码如下:
var objA = new classA('DK');
objA.alertName();//DK
var objB = new classB('DS',20);
objB.alertName();//DS
objB.alertAge();//20
这就是所谓的对象冒充了,另外对象冒充还有另外两种实现的方式,虽然它们的实现手段不一样,但是它们的原理是一样的。
对象冒充实现方法二,使用call方法:
复制代码 代码如下:
var classA = function(name){
this.name = name;
this.alertName = function(){
alert(this.name);
}
}
var classB = function(name,age){
classA.call(this,name);
this.age = age;
this.alertAge = function(){
alert(this.age);
}
}
通过代码也能看出来,第一种方法中我们新建了函数指针指向父类,调用函数,然后将指针删除。而这里我们之间用call方法在this对象下面运行父类的构造函数,实现了同样的目的。另外与call方法相对于的则就是apply方法啦。
对象冒充实现方法三,使用apply方法:
复制代码 代码如下:
var classA = function(name){
this.name = name;
this.alertName = function(){
alert(this.name);
}
}
var classB = function(name,age){
classA.apply(this,new Array(name));
this.age = age;
this.alertAge = function(){
alert(this.age);
}
}
其实大家可以看到,apply方法跟call方法是非常类似的,只不过传递参数是略有不同罢了。
二、原型继承
大家应该对prototype对象有所了解,原型对象上的所有属性和方法将被传递给类的所有实例,所有当我们把父类的所有属性和方法付给子类的prototype对象时也就相当于实现了我们的继承。
子类想获得父类的所有属性和方法,那我们将父类的一个实例直接付给子类的prototype对象,那我们的子类不就相当于获取了父类的所有对象和方法?
代码示例伺候:
复制代码 代码如下:
var classA = function(){
this.name = 'DK';
this.alertName = function(){
alert(this.name);
}
}
var classB = function(name,age){
this.name = name;
this.age = age;
}
classB.prototype = new classA();
classB.prototype.alertAge = function(){
alert(this.age);
}
注意这里的父类的构造函数需要确保没有参数。因为即使有构造参数在实现原型继承的时候你也无法传递=.=!
三、混合继承
顾名思义,混合继承就是前两种方式的混合使用了。
复制代码 代码如下:
var classA = function(name){
this.name = name;
}
classA.prototype.alertName = function(){
alert(this.name);
}
var classB = function(name,age){
classA.call(this,name);
this.age = age;
}
classB.prototype = new classA();
classB.prototype.alertAge = function(){
alert(this.age);
}
使用对象冒充实现了向父类传递参数,同时使用原型继承实现了对公有方法的继承。
说完了这三中继承方式了,下面该说到问题的时候了。
你可能会不解,为什么有了对象冒充,有了原型继承还要再弄出个什么混合继承,对,最重要的也就是这个问题。
1.如果你实际测试一下,你会发现通过对象冒充的方式实现的继承,子类是无法访问到父类的原型链上的方法的。
2.而用原型继承,则会把所有的属性变成共享的属性,如果你同一个子类实现两个实例,你会发现你的所有实例共享所有的属性。
3.但是这肯定是不合适的了。所以就有了混合继承的方式,让属性继续保持私有,同时让子类能够访问父类的原型链的方法。
你可以亲自动手试一下,在对象冒充继承的时候,子类无法访问父类的原型链方法,原型链继承子类的所有实例共享所有父类属性。这里我就不写例子了。
免责声明:本站文章均来自网站采集或用户投稿,网站不提供任何软件下载或自行开发的软件! 如有用户或公司发现本站内容信息存在侵权行为,请邮件告知! 858582#qq.com
稳了!魔兽国服回归的3条重磅消息!官宣时间再确认!
昨天有一位朋友在大神群里分享,自己亚服账号被封号之后居然弹出了国服的封号信息对话框。
这里面让他访问的是一个国服的战网网址,com.cn和后面的zh都非常明白地表明这就是国服战网。
而他在复制这个网址并且进行登录之后,确实是网易的网址,也就是我们熟悉的停服之后国服发布的暴雪游戏产品运营到期开放退款的说明。这是一件比较奇怪的事情,因为以前都没有出现这样的情况,现在突然提示跳转到国服战网的网址,是不是说明了简体中文客户端已经开始进行更新了呢?
更新日志
- 好薇2024《兵哥哥》1:124K黄金母盘[WAV+CUE]
- 胡歌.2006-珍惜(EP)【步升大风】【FLAC分轨】
- 洪荣宏.2014-拼乎自己看【华特】【WAV+CUE】
- 伊能静.1999-从脆弱到勇敢1987-1996精选2CD【华纳】【WAV+CUE】
- 刘亮鹭《汽车DJ玩主》[WAV+CUE][1.1G]
- 张杰《最接近天堂的地方》天娱传媒[WAV+CUE][1.1G]
- 群星《2022年度发烧天碟》无损黑胶碟 2CD[WAV+CUE][1.4G]
- 罗文1983-罗文甄妮-射雕英雄传(纯银AMCD)[WAV+CUE]
- 群星《亚洲故事香港纯弦》雨果UPMAGCD2024[低速原抓WAV+CUE]
- 群星《经典咏流传》限量1:1母盘直刻[低速原抓WAV+CUE]
- 庾澄庆1993《老实情歌》福茂唱片[WAV+CUE][1G]
- 许巍《在别处》美卡首版[WAV+CUE][1G]
- 林子祥《单手拍掌》华纳香港版[WAV+CUE][1G]
- 郑秀文.1997-我们的主题曲【华纳】【WAV+CUE】
- 群星.2001-生命因爱动听电影原创音乐AVCD【MEDIA】【WAV+CUE】