- Original: Classical Inheritance is Obsolete - Chapter 4. Objects - Programming JavaScript Applications
- Translated by: cssmagic
"Those who are unaware they are walking in darkness will never seek the light."
-- Bruce Lee
不知道自己正走在黑暗中的人是永远不会去搜寻光明的。
——李小龙
In "Design Patterns: Elements of Reusable Object Oriented Software", the Gang of Four opened the book with two foundational principles of object oriented design:
在《设计模式:可复用面向对象软件的基础》一书的开头,“四人帮”就推出了面向对象设计的两大基本原则:(译注:该书由四位作者合著,均为国际公认的面向对象软件领域的专家。)
Program to an interface, not an implementation.
面向接口编程,而非面向实现编程。
Favor object composition over class inheritance.
优先使用对象组合,而非继承。
In a sense, the second principle could follow from the first, because inheritance exposes the parent class to all child classes. The child classes are all programming to an implementation, not an interface. Classical inheritance breaks the principle of encapsulation, and tightly couples the child class to its ancestors.
从某种意义上说,第二条规则可由第一条推导出来,因为继承把父类暴露给了所有子类。子类的本质都是面向实现编程,而非面向接口。类继承打破了封装的原则,而且把子类和它的祖先类紧密地耦合起来了。
Think of it this way: Classical inheritance is like Ikea furniture. You have a bunch of pieces that are designed to fit together in a very specific way. If everything goes exactly according to plan, chances are high that you'll come out with a usable piece of furniture, but if anything at all goes wrong or deviates from the preplanned specification, there is little room for adjustment or flexibility. Here's where the analogy (and the furniture, and the software) breaks down: The design is in a constant state of change.
不妨从这个角度来想想: 类继承类似于宜家的家具。你有一堆零件,它们天生需要以一种非常特定的方式来组装。如果每个零件都严格按计划组装,那么你应该会得到一件可用的家具;但如果任何零件装错了,或偏离了说明书的规定,那就没有多少灵活调整的空间了。这个比喻(家具或是软件)崩溃的原因在于:这种设计需要面对不断的变数。
Composition is more like Lego blocks. The various pieces aren't designed to fit with any specific piece. Instead, they are all designed to fit together with any other piece, with few exceptions.
而组合更像是乐高积木。各式各样的零件并不只能与指定的零件组合。相反,每一块积木都被设计为可以与其它零件任意组合,几乎没有例外。
When you design for classical inheritance, you design a child class to inherit from a specific parent class. The specific parent class name is usually hard coded right in the child class, with no mechanism to override it. Right from the start, you're boxing yourself in -- limiting the ways that you can reuse your code without rethinking its design at a fundamental level.
当你在设计类继承时,你会从一个特定的父类继承出一个子类。这个特定的父类的名称通常会在子类中写死,而没有任何机制可以覆盖它。而从一开始,你就在跟自己搏斗——你限制了重用代码的方式,而没有在一个基本层面去反思它的设计。
When you design for composition, the sky is the limit. As long as you can successfully avoid colliding with properties from other source objects, objects can be composed and reused virtually any way you see fit. Once you get the hang of it, composition affords a tremendous amount of freedom compared to classical inheritance. For people who have been immersed in classical inheritance for years, and learn how to take real advantage of composition (specifically using prototypal techniques), it is literally like walking out of a dark tunnel into the light and seeing a whole new world of possibilities open up for you.
当你在设计组合方式的时候,那就是一片海阔天空。只要你可以成功地避免其它源对象引入的属性冲突,对象实际上就可以以任何你认为合适的方式进行组合和重用。一旦你掌握了其中的要领,相对于类继承,组合将赋予你极大的自由。对于那些在类继承中已经沉浸多年的人来说,学习如何从组合中受益(尤其是使用原型方面的技巧),真的像是从一条黑暗的地道中走向光明,发现一个全新的世界正向你敞开大门。
Back to "Design Patterns". Why is the seminal work on Object Oriented design so distinctly anti-inheritance? Because inheritance causes several problems:
回到“设计模式”。为什么这本面向对象领域的著作会如此旗帜鲜明地反对继承?因为继承会导致以下问题:(译注:“面向对象”即 Object Oriented,以下简称 OO。)
Tight coupling. Inheritance is the tightest coupling available in OO design. Descendant classes have an intimate knowledge of their ancestor classes.
强耦合。在 OO 设计中,继承是所能找到的最强的耦合方式。后代类对它们的祖先类了如指掌。
Inflexible hierarchies. Single parent hierarchies are rarely capable of describing all possible use cases. Eventually, all hierarchies are "wrong" for new uses -- a problem that necessitates code duplication.
不灵活的层级系统。单个父类层级很难描述所有应用场景的可能性。最终,所有层级对新用法来说都是“错误”的——不得不引入代码重复的问题。
Multiple inheritance is complicated. It's often desirable to inherit from more than one parent. That process is inordinately complex and its implementation is inconsistent with the process for single inheritance, which makes it harder to read and understand.
多重继承十分复杂。从多个父类继承有时会很诱人。这种方法非常复杂,并且它的实现与单继承的方式不一致,这将导致它难于阅读和理解。
Brittle architecture. Because of tight coupling, it's often difficult to refactor a class with the "wrong" design, because much existing functionality depends on the existing design.
脆弱的架构。由于强耦合的存在,通常很难对一个使用了“错误”设计的类进行重构,因为有太多既有功能依赖这些既有设计。
The Gorilla / Banana problem. Often there are parts of the parent that you don't want to inherit. Subclassing allows you to override properties from the parent, but it doesn't allow you to select which properties you want to inherit.
大猩猩与香蕉问题。父类总会有某些部分是你不想继承的。子类允许你覆盖父类的属性,但它不允许你选择哪些属性是你想继承的。
These problems are summed up nicely by Joe Armstrong in "Coders at Work", by Peter Siebel:
这些问题都已经被 Joe Armstrong 很好地总结在了《Coders at Work》一书中,该书由 Peter Siebel 联合执笔。
“The problem with object-oriented languages is they've got all this implicit environment that they carry around with them. You wanted a banana but what you got was a gorilla holding the banana and the entire jungle.”
面向对象语言与生俱来的问题就是它们与生俱来的这一整个隐性环境。你想要一根香蕉,但你得到的是一头手里握着香蕉的大猩猩,以及整个丛林。
-- Joe Armstrong
Inheritance works beautifully for a short time, but eventually the app architecture becomes arthritic. When you've built up your entire app on a foundation of classical inheritance, the dependencies on ancestors run so deep that even reusing or changing trivial amounts of code can turn into a gigantic refactor. Deep inheritance trees are brittle, inflexible, and difficult to extend.
在短时间内,继承会工作得很完美,但最终,应用的架构会变得僵硬迟缓。当你已经在一个类继承体系上建立起你的整个应用之后,由于祖先类的依赖关系是如此之深,以致于重用或改变一些细节末节的代码都将导致一场大规格的重构。深度继承树是脆弱的、不灵活的、难于扩展的。
More often than not, what you wind up with in a mature classical OO application is a range of possible ancestors to inherit from, all with slightly different but often similar configurations. Figuring out which to use is not straightforward, and you soon have a haphazard collection of similar objects with unexpectedly divergent properties. Around this time, people start throwing around the word "rewrite" as if it's an easier undertaking than refactoring the current mess.
多半时候,在一个成熟的基于类的 OO 应用程序中,你最终得到的是一堆可以用来继承的祖先类,它们都有着细微的差别,但又常常具有相似的配置。找出一个适用的祖先类并不容易,而且你很快会拥有一堆杂乱无章的相似对象,这些对象又拥有一堆杂乱无章的属性。到了这个时候,人们开始抛出“重写”这个词,仿佛这样会比重构眼下这一团槽更容易些。
Many of the patterns in the GoF book were designed specifically to address these well-known problems. In many ways, the book itself can be read as a critique of the shortcomings of most classical OO languages, along with the accompanying lengthy work-arounds. In short, patterns point out deficiencies in the language. You can reproduce all of the GoF patterns in JavaScript, but before you start using them as blueprints for your JavaScript code, you'll want to get a good handle on JavaScript's prototypal and functional capabilities.
“四人帮”书里的很多模式都特别适合讲解这些著名的问题。在很多方面,这本书读起来就像是对大多数基于类的 OO 语言缺陷的一种批判,同时书中也提供了冗长的变通解决方案。简单来说,设计模式指出了语言中的短板。你可以用 JavaScript 重现书中的所有模式,但在你准备以它们为基础来构建你的 JavaScript 代码,你不妨先掌握好 JavaScript 的原型式和函数式编程能力。
For a long time, many people were confused about whether JavaScript is truly object oriented, because they felt that it lacked features from other OO languages. Setting aside the fact that JavaScript handles classical inheritance with less code than most class-based languages, coming to JavaScript and asking how to do classical inheritance is like picking up a touch screen mobile phone and asking where the rotary dial is. Of course people will be amused when the next thing out of your mouth is, "if it doesn't have a rotary dial, it's not a telephone!"
在很长一段时间内,很多人都对 JavaScript 是否是一门真正的 OO 语言感到疑惑,因为他们感觉它缺少其它 OO 语言的一些特性。暂不提 JavaScript 只需要(与大多数基于类的语言相比)更少的代码就可以搞定类继承;刚接触 JavaScript 就问如何实现类继承,就好像捡起一部触屏手机然后问别人它的拨号转盘在哪里。当然,如果你回答“它没有拨号转盘,它不是电话机”,那些人也许会被逗乐。
JavaScript can do most of the OO things you're accustomed to in other languages, such as inheritance, data privacy, polymorphism, and so on. However, JavaScript has many native capabilities that make some classical OO features and patterns obsolete. It's better to stop asking "how do I do classical inheritance in JavaScript?", and start asking, "what cool new things does JavaScript enable me to do?"
JavaScript 几乎可以实现所有其它语言所能做到的 OO 行为,比如继承、数据私有化、多态等等。然而,JavaScript 原生具备的很多能力就足以让一些基于类的 OO 特性和模式变得过时。所以最好别再问“怎样在 JavaScript 中实现类继承”,而应该问“我在 JavaScript 中可以做哪些不一样的、超酷的事情?”
[!]
I wish I could tell you that you'll never have to deal with classical inheritance in JavaScript. Unfortunately, because classical inheritance is easy to mimic in JavaScript, and many people come from class-based programming backgrounds, there are several popular libraries which feature classical inheritance prominently, including Backbone.js, which you'll have a chance to explore soon. When you do encounter situations in which you're forced to subclass by other programmers, keep in mind that inheritance hierarchies should be kept as small as possible. Avoid subclassing subclasses, remember that you can mix and match different code, reuse styles, and things will go more smoothly.
[!]
但愿你可以明白,你永远不需要在 JavaScript 中使用类继承模式。但不幸的是,由于类继承在 JavaScript 中很容易模似,而且有太人来自于基于类的编程背景,导致很一些流行的类库特意引入了类继承的特性,这其中包括 Backbone.js,我们后面会有机会详细了解它。如果你确实遇到不得不使用其它程序员写的子类的情况,请牢记继承层级应该控制得尽可能少。要避免创建子类的子类,别忘了你可以混合并匹配不同的代码、重用样式,然后事情会变得更加顺利。