Javascript 继承机制的实现


Posted in Javascript onAugust 12, 2009

选定基类后,就可以创建它的子类了。是否使用基类完全由你决定。有时,你可能想创建一个不能直接使用的基类,它只是用于给子类提供通用的函数。在这种情况下,基类被看作抽象类。

尽管ECMAScript并没有像其他语言那样严格地定义抽象类,但有时它的确会创建一些不允许使用的类。通常,我们称这种类为抽象类。

创建的子类将继承超类的所有属性和方法,包括构造函数及方法的实现。记住,所有属性和方法都是公用的,因此子类可直接访问这些方法。子类还可添加超类中没有的新属性和方法,也可以覆盖超类中的属性和方法。

4.2.1 继承的方式

和其他功能一样,ECMAScript中实现继承的方式不止一种。这是因为JavaScript中的继承机制并不是明确规定的,而是通过模仿实现的。这意味着所有的继承细节并非由解释程序处理。作为开发者,你有权决定最适用的继承方式。

1. 对象冒充

构想原始的ECMAScript时,根本没打算设计对象冒充(object masquerading)。它是在开发者开始理解函数的工作方式,尤其是如何在函数环境中使用this关键字后才发展出来的。

其原理如下:构造函数使用this关键字给所有属性和方法赋值(即采用类声明的构造函数方式)。因为构造函数只是一个函数,所以可使ClassA的构造函数成为ClassB的方法,然后调用它。ClassB就会收到ClassA的构造函数中定义的属性和方法。例如,用下面的方式定义ClassA和ClassB:

Javascript 继承机制的实现

还记得吗?关键字this引用的是构造函数当前创建的对象。不过在这个方法中,this指向的是所属的对象。这个原理是把ClassA作为常规函数来建立继承机制,而不是作为构造函数。如下使用构造函数ClassB可以实现继承机制:

Javascript 继承机制的实现

Javascript 继承机制的实现在这段代码中,为ClassA赋予了方法newMethod(记住,函数名只是指向它的指针)。然后调用该方法,传递给它的是ClassB构造函数的参数sColor。最后一行代码删除了对ClassA的引用,这样以后就不能再调用它。

所有的新属性和新方法都必须在删除了新方法的代码行后定义。否则,可能会覆盖超类的相关属性和方法:

Javascript 继承机制的实现

为证明前面的代码有效,可以运行下面的例子:

Javascript 继承机制的实现

有趣的是,对象冒充可以支持多重继承。也就是说,一个类可以继承多个超类。用UML表示的多继承机制如图4-2所示。

Javascript 继承机制的实现

图  4-2

例如,如果存在两个类ClassX和ClassY,ClassZ想继承这两个类,可以使用下面的代码:

Javascript 继承机制的实现Javascript 继承机制的实现

Javascript 继承机制的实现

这里存在一个弊端,如果ClassX和ClassY具有同名的属性或方法,ClassY具有高优先级,因为继承的是最后的类。除这点小问题之外,用对象冒充实现多继承机制轻而易举。

由于这种继承方法的流行,ECMAScript的第三版为Function对象加入了两个新方法,即call()和apply()。

2. call()方法

call()方法是与经典的对象冒充方法最相似的方法。它的第一个参数用作this的对象。其他参数都直接传递给函数自身。例如:

Javascript 继承机制的实现

Javascript 继承机制的实现

在这个例子中,函数sayColor()在对象外定义,即使它不属于任何对象,也可以引用关键字this。对象obj的color属性等于"red"。调用call()方法时,第一个参数是obj,说明应该赋予sayColor()函数中的this关键字值是obj。第二个和第三个参数是字符串。它们与sayColor()函数中的参数prefix和suffix匹配,最后生成的消息"The color is red, a very nice color indeed"将被显示出来。

要与继承机制的对象冒充方法一起使用该方法,只需将前三行的赋值、调用和删除代码替换即可:

Javascript 继承机制的实现

Javascript 继承机制的实现这里,想让ClassA中的关键字this等于新创建的ClassB对象,因此this是第一个参数。第二个参数sColor对两个类来说都是唯一的参数。

3. apply()方法

apply()方法有两个参数,用作this的对象和要传递给函数的参数的数组。例如:

Javascript 继承机制的实现

这个例子与前面的例子相同,只是现在调用的是apply()方法。调用apply()方法时,第一个参数仍是obj,说明应该赋予sayColor()中的this关键字值是obj。第二个参数是由两个字符串构成的数组,与sayColor()的参数prefix和suffix匹配。生成的消息仍是"The color is red, a very nice color indeed",将被显示出来。

该方法也用于替换前三行的赋值、调用和删除新方法的代码:

Javascript 继承机制的实现

Javascript 继承机制的实现

同样的,第一个参数仍是this。第二个参数是只有一个值color的数组。可以把ClassB的整个arguments对象作为第二个参数传递给apply()方法:

Javascript 继承机制的实现

Javascript 继承机制的实现当然,只有超类中的参数顺序与子类中的参数顺序完全一致时才可以传递参数对象。如果不是,就必须创建一个单独的数组,按照正确的顺序放置参数。此外,还可使用call()方法。

4. 原型链

继承这种形式在ECMAScript中原本是用于原型链的。上一章介绍了定义类的原型方式。原型链扩展了这种方式,以一种有趣的方式实现继承机制。

在上一章中学过,prototype对象是个模板,要实例化的对象都以这个模板为基础。总而言之,prototype对象的任何属性和方法都被传递给那个类的所有实例。原型链利用这种功能来实现继承机制。

如果用原型方式重定义前面例子中的类,它们将变为下列形式:

Javascript 继承机制的实现

原型链的神奇之处在于突出显示的代码行。这里,把ClassB的prototype属性设置成ClassA的实例。这很有意义,因为想要ClassA的所有属性和方法,但又不想逐个将它们赋予ClassB的prototype属性。还有比把ClassA的实例赋予prototype属性更好的方法吗?

注意,调用ClassA的构造函数时,没有给它传递参数。这在原型链中是标准做法。要确保构造函数没有任何参数。

与对象冒充相似,子类的所有属性和方法都必须出现在prototype属性被赋值后,因为在它之前赋值的所有方法都会被删除。为什么?因为prototype属性被替换成了新对象,添加了新方法的原始对象将被销毁。所以,为ClassB类添加name属性和sayName()方法的代码如下:

Javascript 继承机制的实现Javascript 继承机制的实现

Javascript 继承机制的实现

可通过运行下面的例子测试这段代码:

Javascript 继承机制的实现

此外,在原型链中,instanceof运算符的运行方式也很独特。对ClassB的所有实例,instanceof为ClassA和ClassB都返回true。例如:

Javascript 继承机制的实现

在ECMAScript的弱类型世界中,这是极其有用的工具,不过使用对象冒充时不能使用它。

原型链的弊端是不支持多重继承。记住,原型链会用另一类型的对象重写类的prototype属性。

5. 混合方式

这种继承方式使用构造函数定义类,并未使用任何原型。对象冒充的主要问题是必须使用构造函数方式(如上一章中学到的),这不是最好的选择。不过如果使用原型链,就无法使用带参构造函数了。开发者该如何选择呢?答案很简单,两者都用。

在前一章,你学过创建类的最好方式是用构造函数方式定义属性,用原型方式定义方法。这种方式同样适用于继承机制,用对象冒充继承构造函数的属性,用原型链继承prototype对象的方法。用这两种方式重写前面的例子,代码如下:

Javascript 继承机制的实现Javascript 继承机制的实现

Javascript 继承机制的实现

在此例子中,继承机制由两行突出显示的代码实现。在第一行突出显示的代码中,在ClassB构造函数中,用对象冒充继承ClassA类的sColor属性。在第二行突出显示的代码中,用原型链继承ClassA类的方法。由于这种混合方式使用了原型链,所以instanceof运算符仍能正确运行。

下面的例子测试了这段代码:

Javascript 继承机制的实现

Javascript 相关文章推荐
用js来解决ajax读取页面乱码
Nov 28 Javascript
Js从头学起(基本数据类型和引用类型的参数传递详细分析)
Feb 16 Javascript
Jquery多选下拉列表插件jquery multiselect功能介绍及使用
May 24 Javascript
下拉列表select 由左边框移动到右边示例
Dec 04 Javascript
jquery form表单获取内容以及绑定数据
Feb 24 Javascript
jquery把int类型转换成字符串类型的方法
Oct 07 Javascript
JS实现获取来自百度,Google,soso,sogou关键词的方法
Dec 21 Javascript
原生JS与jQuery编写简单选项卡
Oct 30 jQuery
Vue CLI 2.x搭建vue(目录最全分析)
Feb 27 Javascript
微信小程序tab切换可滑动切换导航栏跟随滚动实现代码
Sep 04 Javascript
微信小程序pinker组件使用实现自动相减日期
May 07 Javascript
微信小程序实现打卡签到页面
Sep 21 Javascript
Javascript 继承实现例子
Aug 12 #Javascript
JQuery jsonp 使用示例代码
Aug 12 #Javascript
JavaScript 权威指南(第四版) 读书笔记
Aug 11 #Javascript
Javascript+XMLHttpRequest+asp.net无刷新读取数据库数据
Aug 09 #Javascript
javascript 当前日期加(天、周、月、年)
Aug 09 #Javascript
jQuery 解析xml文件
Aug 09 #Javascript
ext 代码生成器
Aug 07 #Javascript
You might like
咖啡风味 世界咖啡主要分布分布 咖啡的生长要求
2021/03/06 新手入门
PHP学习笔记 用户注册模块用户类以及验证码类
2011/09/20 PHP
基于session_unset与session_destroy的区别详解
2013/06/03 PHP
php自定义函数实现汉字转换utf8编码的方法
2016/09/29 PHP
PHP实现根据数组的值进行分组的方法
2017/04/20 PHP
[原创]php实现数组按拼音顺序排序的方法
2017/05/03 PHP
Laravel Eloquent ORM 实现查询表中指定的字段
2019/10/17 PHP
Laravel框架数据库迁移操作实例详解
2020/04/06 PHP
使用jQuery内容过滤选择器选择元素实例讲解
2013/04/18 Javascript
Jquery实现点击按钮,连续地向textarea中添加值的实例代码
2014/03/08 Javascript
JavaScript中的迭代器和生成器详解
2014/10/29 Javascript
基于jquery实现的仿优酷图片轮播特效代码
2016/01/13 Javascript
jQuery点击弹出层弹出模态框点击模态框消失代码分享
2017/01/21 Javascript
微信小程序 判断手机号的实现代码
2017/04/19 Javascript
Angular2实现组件交互的方法分析
2017/12/19 Javascript
Vuex 在Vue 组件中获得Vuex 状态state的方法
2018/08/27 Javascript
详解React 的几种条件渲染以及选择
2018/10/23 Javascript
jquery实现的分页显示功能示例
2019/08/23 jQuery
Python中请使用isinstance()判断变量类型
2014/08/25 Python
pyqt5对用qt designer设计的窗体实现弹出子窗口的示例
2019/06/19 Python
Python数据可视化:箱线图多种库画法
2019/11/06 Python
jupyter notebook运行命令显示[*](解决办法)
2020/05/18 Python
结合CSS3的新特性来总结垂直居中的实现方法
2016/05/30 HTML / CSS
营销总经理的岗位职责
2013/12/15 职场文书
小学生学习雷锋倡议书
2014/05/15 职场文书
会计系毕业生求职信
2014/05/28 职场文书
环卫工人先进事迹材料
2014/06/02 职场文书
国际经济与贸易专业求职信
2014/07/10 职场文书
无犯罪记录证明
2014/09/19 职场文书
环境工程专业毕业生求职信
2014/09/30 职场文书
运动会通讯稿600字
2015/07/20 职场文书
关于五一放假的通知
2015/08/18 职场文书
2016党员学习《反对自由主义》心得体会
2016/01/22 职场文书
六年级语文教学反思
2016/03/03 职场文书
详解Vue的sync修饰符
2021/05/15 Vue.js
使用tensorflow 实现反向传播求导
2021/05/26 Python