《JavaScript应用程序设计》一一3.1 过时的类继承

简介:

本节书摘来华章计算机出版社《JavaScript应用程序设计》一书中的第3章,第3.1节,作者:Eric Elliott 更多章节内容可以访问云栖社区“华章计算机”公众号查看。

3.1 过时的类继承

不知道自己正走在黑暗中的人是永远不会去搜寻光明的。
——李小龙
在《设计模式:可复用面向对象软件的基础》一书的首章,“四人帮”向我们介绍了面向对象设计的两大基本原则:

  1. 面向接口编程,而非面向实现编程。
  2. 优先使用(对象)组合,而非(类)继承。
    从某种角度上说,第二条原则是从第一条中派生而来,因为父类通过“继承”将其内部结构直接暴露给了所有子类,而子类从本质上来说都是在面向实现编程,并非面向接口。类继承违反了代码封装的原则,它使得子类与其祖先之间产生了紧耦合。

我们可以将类继承比作是宜家的家具,假设你买来一堆家具零部件,它们需要按照一定方式组装在一起,如果每个零部件都严格参照说明书中的描述来组装,那么大部分情况下你的家具不会有问题;但如果碰巧某个零部件本身存在缺陷,或者在组装时没有按照说明书中的操作来做,则家具可能就此就报废了。如果把这个例子放在软件开发领域中会得到这样一条结论:代码设计总是在不停地变化。
组合则看起来更像是乐高积木,没有谁规定说积木非得要搭建成什么样子,所有积木之间都可以做自由组合,从而形成不同形状、不同大小的物体(玩具)。
当你在为类继承做代码设计时,你总是在思考如何从具体父类中派生出子类。父类大都会将名称直接硬编码进子类,完全不给你做重写的机会。使用类继承等于一开始就将自己囚禁了起来,限制了代码复用性不说,关键会导致你不会从一个基本层面上反思代码设计。
而当你在为组合做代码设计时,你只需要留心多个对象间的属性是否会有冲突就可以了,总体上来说整个设计的过程中不会有太多的约束,对象可以根据你的设计做任意组合与重用。一旦你掌握了组合的要领,相比类继承来说,在代码设计上将获得更大的自由度。对于那些常年在类继承体系下工作的程序员来说,学习与掌握组合(尤其是使用原型技术)就像是在漆黑狭窄的隧道中,发现远方的一束光亮一样,迎接他的是一个充满了各种可能性的崭新世界。
我们先暂时回到《设计模式》这本书中来,为什么这本面向对象领域的著作会如此旗帜鲜明地反对继承?因为继承会引发下列问题:
对象或模块间的强耦合:
在面向对象程序设计中,继承是最容易导致强耦合的代码复用方式,因为派生类对其祖先类的一切都十分了解。
继承层级结构缺乏灵活性:
单一父类的继承层级结构本身就很难覆盖到所有功能用例,而且对新加入的功能用例的适配性也很差,最终导致代码彼此重复。
多重继承同时带来了代码复杂度:
让子类继承多个父类,这从代码复用的角度看来十分具有诱惑性,但它实现的过程不仅非常复杂,而且与单继承有很大的差异性,这常常也是多重继承的代码结构非常晦涩的原因。
架构变得极为不稳定:
一般情况下,由于强耦合的存在,使得你很难对一个采用了“错误”设计的类进行重构,因为有太多的现有功能依赖它。
大猩猩与香蕉问题的产生:
在父类中,总会有一些特性是你不想让子类继承的,子类可以重写父类的部分特性,但它决定不了哪些特性是你真正想要从父类中继承的。
《Coders at Work》一书对绝大部分类继承所引发的问题做了总结梳理,该书由Joe Armstrong与Peter Siebel联合执笔(http://www.codersatwork.com/)
面向对象语言的问题,就在于它本身所持有的隐性语言环境。好比说你想要一根香蕉,它却把握着香蕉的大猩猩扔给你,而有时甚至会是整个森林。
继承只能够带来一时的效率提升,而到了后期应用的架构,就像是患了关节炎一样,变得行动迟缓起来。当你将整个应用搭建在类继承体系上时,任何对代码的细微修改都可能导致大范围的代码重构,这是因为上层祖先类间的依赖关系往往是非常紧密的。继承树过深,会导致代码丧失灵活性、易碎且难于扩展。
一般来说,在一个严重依赖于类继承的应用程序中,会有大量祖先类的存在,这些祖先类相互间略有差异,但是配置却很相像,这使得分辨它们各自的用途成了一件难事,而且应用很快便被一群看起来极为相似,行为却十分迥异的实例对象所充斥。“重写”一词多半就是在这个时候被抛出来的,人们仿佛觉得这样做,会比代码重构来的要便捷。
《设计模式》一书中的绝大部分内容其实就是针对上述这些问题的解决方案,但是这些方案有时会显得较为烦琐,反而给程序引入了更多的代码复杂度。有趣的是,这本书的内容读起来会让人感觉像是对所有面向对象语言的一次集体揭短。虽然说你可以将书中的所有模式直接搬进JavaScript中,不过建议你在使用它们之前,最好先把JavaScript中原型与函数的概念了解清楚。
很多人一直以来都不太明白JavaScript究竟是不是一门真正的面向对象语言,因为他们认为,JavaScript相比其他面向对象语言来说,总有些缺胳膊少腿的感觉。在这里,我们抛开JavaScript 可以模拟类继承的能力(与多数基于类的语言相比代码要更为精简)不谈,如果你刚接触JavaScript就开始寻思着如何引入类继承,就好比是拿着智能触屏手机问别人拨号转盘在哪儿一样,此时如果你一本正经地告诉别人:“它根本就不是电话机,因为它没有拨号转盘”,人们会认为你在搞笑。
如继承、数据隐蔽性、多态等,这些原本出自于其他语言的面向对象特性,在JavaScript中都可以被模拟,但JavaScript与生俱来的一些语言特性反而会让人们觉得,他们可以在JavaScript中完全废弃这些基于类继承的编码模式。所以别再去纠结诸如“我如何在JavaScript中实现类继承”之类的问题了,反而你更应该去思考:“JavaScript能够带给我哪些全新的特性?”。
警告: 我希望你能明白一点,你永远不需要在JavaScript中使用类继承。说是这么说,但由于类继承在JavaScript 中非常容易被模拟,加之有很多人都有传统面向对象语言的编程背景,所以导致市面上许多类库都有意引入了类继承的概念,这其中就包括 Backbone.js,我会在后续章节对它进行介绍。如果你在编码中不得不使用类继承,那么请严格控制继承的层级深度。有很多代码复用方式能够代替继承,它们往往具有更好的延展性。

相关文章
|
3月前
|
设计模式 JavaScript 前端开发
在JavaScript中,继承是一个重要的概念,它允许我们基于现有的类(或构造函数)创建新的类
【6月更文挑战第15天】JavaScript继承促进代码复用与扩展,创建类层次结构,但过深的继承链导致复杂性增加,紧密耦合增加维护成本,单继承限制灵活性,方法覆盖可能隐藏父类功能,且可能影响性能。设计时需谨慎权衡并考虑使用组合等替代方案。
41 7
|
11天前
|
JavaScript 前端开发
JavaScript基础知识-构造函数(也称为"类")定义
本文介绍了JavaScript中构造函数(也称为“类”)的定义和使用方法。
20 1
JavaScript基础知识-构造函数(也称为"类")定义
|
3天前
|
JavaScript 前端开发
JS中Promise的类式实现写法
JS中Promise的类式实现写法
|
13天前
|
开发者 图形学 iOS开发
掌握Unity的跨平台部署与发布秘籍,让你的游戏作品在多个平台上大放异彩——从基础设置到高级优化,深入解析一站式游戏开发解决方案的每一个细节,带你领略高效发布流程的魅力所在
【8月更文挑战第31天】跨平台游戏开发是当今游戏产业的热点,尤其在移动设备普及的背景下更为重要。作为领先的游戏开发引擎,Unity以其卓越的跨平台支持能力脱颖而出,能够将游戏轻松部署至iOS、Android、PC、Mac、Web及游戏主机等多个平台。本文通过杂文形式探讨Unity在各平台的部署与发布策略,并提供具体实例,涵盖项目设置、性能优化、打包流程及发布前准备等关键环节,助力开发者充分利用Unity的强大功能,实现多平台游戏开发。
33 0
|
22天前
|
JavaScript 前端开发 开发者
揭开JavaScript的神秘面纱:原型链背后隐藏的继承秘密
【8月更文挑战第23天】原型链是JavaScript面向对象编程的核心特性,它使对象能继承另一个对象的属性和方法。每个对象内部都有一个[[Prototype]]属性指向其原型对象,形成链式结构。访问对象属性时,若当前对象不存在该属性,则沿原型链向上查找。
23 0
|
30天前
|
JavaScript 前端开发
JS的6种继承方式
JS的6种继承方式
|
30天前
|
JavaScript 前端开发
记录Javascript数组类练习
记录Javascript数组类练习
|
3月前
|
设计模式 JavaScript 前端开发
【JavaScript】深入浅出JavaScript继承机制:解密原型、原型链与面向对象实战攻略
JavaScript的继承机制基于原型链,它定义了对象属性和方法的查找规则。每个对象都有一个原型,通过原型链,对象能访问到构造函数原型上的方法。例如`Animal.prototype`上的`speak`方法可被`Animal`实例访问。原型链的尽头是`Object.prototype`,其`[[Prototype]]`为`null`。继承方式包括原型链继承(通过`Object.create`)、构造函数继承(使用`call`或`apply`)和组合继承(结合两者)。ES6的`class`语法是语法糖,但底层仍基于原型。继承选择应根据需求,理解原型链原理对JavaScript面向对象编程至关重要
71 7
【JavaScript】深入浅出JavaScript继承机制:解密原型、原型链与面向对象实战攻略
|
1月前
|
设计模式 JavaScript 前端开发
js对原型和继承的理解
了解JavaScript中原型和继承的概念对于编写优雅高效的代码、理解库和框架的内部机制以及执行高级设计模式都有着重要的意义。
36 0
|
3月前
|
JavaScript 前端开发
JavaScript进阶-原型链与继承
【6月更文挑战第18天】JavaScript的原型链和继承是其面向对象编程的核心。每个对象都有一个指向原型的对象链,当查找属性时会沿着此链搜索。原型链可能导致污染、效率下降及构造函数与原型混淆的问题,应谨慎扩展原生原型、保持原型结构简洁并使用`Object.create`或ES6的`class`。继承方式包括原型链、构造函数、组合继承和ES6的Class继承,需避免循环引用、方法覆盖和不当的构造函数使用。通过代码示例展示了这两种继承形式,理解并有效利用这些机制能提升代码质量。
54 5