书籍作者:Robert C. Martin | ISBN:9787121347962 |
书籍语言:简体中文 | 连载状态:全集 |
电子书格式:pdf,txt,epub,mobi,azw3 | 下载次数:4633 |
创建日期:2021-02-14 | 发布日期:2021-02-14 |
运行环境:PC/Windows/Linux/Mac/IOS/iPhone/iPad/Kindle/Android/安卓/平板 |
《架构整洁之道》是创造“Clean神话”的Bob大叔在架构领域的登峰之作,围绕“架构整洁”这一重要导向,系统地剖析其缘起、内涵及应用场景,涵盖软件研发完整过程及所有核心架构模式。《架构整洁之道》分为6部分,第1部分纲领性地提出软件架构设计的终目标,描述软件架构设计的重点与模式;第2~4部分从软件开发中三个基础编程范式的定义和特征出发,进一步描述函数、组件、服务设计与实现的定律,以及它们是如何有效构建软件系统的整体架构的;第5部分从整洁架构的定义开始,详细阐述软件架构设计过程中涉及的方方面面,包括划分内部组件边界、应用常见设计模式、避开错误、降低成本、处理特殊情况等,并以实战案例将内容有机整合起来;第6部分讲述具体实现细节;附录则透过作者数十年的软件从业经历再次印证《架构整洁之道》的观点。
对于每一位软件研发从业人员——无论从事的是具体编码实现、架构设计,还是软件研发管理,《架构整洁之道》都是不可或缺的。
Robert C. Martin,Object Mentor公司总裁,面向对象设计、模式、UML、敏捷方法学和极限编程领域的资深顾问。他是Designing Object-Oriented C++ Applications Using the Booch Method 以及 Jolt 获奖图书 Agile Software Development, Principles,Palterns,and Practices(中译版《敏捷软件开发:原则、模式与实践》)《代码整洁之道》等畅销书作者。
译者简介
孙宇聪:曾在谷歌工作多年,任谷歌高级SRE(Senior Site Reliblity Engineer),前Coding.net 技术负责人。
善用软件架构的通用法则,即可显著提升开发者在所有软件系统全生命周期内的生产力。如今,传奇软件匠师Robert C. Martin(Bob大叔),携畅销书Clean Code与The Clean Coder所获巨大成功之威,向我们深刻揭示了这些法则并亲授运用之道。
Martin在《架构整洁之道》中远不只是在为我们提供选项,他几乎是在将软件世界中横跨半个世纪的各种架构类型的经验倾囊相授,目的是让读者既能阅尽所有架构选型,又可通晓其如何决定成败。Martin也的确不负厚望,本书中充满了直接而有效的解决方案,以供读者应对自己面临的真正挑战——那些或最终成就或彻底破坏项目的挑战。
《架构整洁之道》不可不读,无论读者是现任的还是将来的软件架构师、系统分析师、系统设计师或软件项目经理,或是身负将他人设计落地重任的开发人员,这本书都可以让你们受益匪浅。
序言
软件架构(architecture)究竟是什么?
不论从哪个角度分析软件系统,都不可能面面俱到。如果从架构学角度来分析,在一定程度上能够做到抓大放小,把握住重点,但是也不可避免地会错失某些重要的细节信息。
软件架构学关注的的一个重点是组织结构(structure)。不管是讨论组件(Component)、类(Class)、函数(Function)、模块(Module),还是层级(Layer)、服务(Service)以及微观与宏观的软件开发过程,软件的组织结构都是我们的主要关注点。但是真实世界中的许多软件项目并不完全按照我们的信念和愿望生长——它们就像超大型国企那样,层层嵌套,缠绕成一团乱麻 。有的时候真的很难相信,软件项目的组织结构性也能像物理建筑那样一目了然,层次清晰。
物理建筑,不管其地基是石头还是水泥,形状是高大还是宽阔,风格是气势恢宏还是小巧玲珑,其组织结构都一目了然。物理建筑的组织结构必须遵守“受重力”这一自然规律,同时还要符合建筑材料自身的物理特性。软件项目则没有定律可以遵循。另外,物理建筑是用砖头、水泥、木头、钢铁或者玻璃等标准材料建成的,而大型软件项目往往是由小的软件组件构成的,这些软件组件又是由更小的软件组件构成的,层层堆叠,无穷无尽。
所以,当讨论软件架构时,要特别注意软件项目是具有递归(recursive)和分形(fractal)特点的,最终都要由一行行的代码组成。脱离了一行行的代码,脱离了具体的细节设计,架构设计就无从谈起。大型物理建筑通常可以用比例模型分层描述细节信息,但是软件项目内部结构是很难用模型分层描述的。软件项目也具有内部结构,但是其结构无论从数量上还是多样性上来说,都远远超过了物理建筑的结构。可以不夸张地说,软件开发比修建物理建筑需要更长、更专注的设计过程,软件架构师应该比建筑架构师更懂架构!
比例模型是深入人心的展示方式,但是不管某个PowerPoint图表中的彩色方块多么好看,多么简单易懂,它也无法完全代表一个软件的架构。它只能是该软件的架构的一个视图,而非全部。软件的架构并没有固定的展现形式,你所看到的每一个视图的背后都是架构师所做的层层抉择。一个视图包含了哪些部分,排除了哪些部分;用特殊形状和颜色强调了哪些部分,又有哪些部分被泛泛地一笔带过,甚至直接忽略,这些都是这个视图本身的特性。然而,每个视图都是对的,它们往往并没有优劣之分。
虽然软件无法很好地用比例模型展示,但它还是要在现实世界中运行的。在设计软件架构的过程中,我们必须理解和遵守现实的约束条件。CPU速度和网络带宽往往在很大程度上决定了系统的性能,而内存和存储空间的大小也会大幅影响代码的设计野心。
女士,这就是爱情的穷凶极恶之处,人的意愿是无穷的,而实际行动却处处受限。人的欲望是无止境的,行为却不得不遵从现实的限制。
——威廉·莎士比亚
人类的整个经济活动都是存在于现实世界中的,所以我们可以利用现实世界的一些准则来衡量和推理软件开发过程中那些不好量化和物化的因素。
软件架构是系统设计过程中的重要设计决定的集合,可以通过变更成本来衡量每个设计决定的重要程度。
——Grady Booch
需要付出的时间、金钱和人力成本是区分软件架构规模大小的衡量标准,也可以用来区分架构设计和细节设计。同时,我们还可以依据这个信息来判断某个特定架构设计是好还是坏:一个好的架构,不仅要在某一特定时刻满足软件用户、开发者和所有者的需求,更要在一段时间内持续满足他们的后续需求。
如果你觉得好架构的成本太高,那你可以试试选择差的架构加上返工重来的成本。
——Brian Foote 和 Joseph Yoder
一个系统的常规变更不应该是成本高昂的,也不应该需要难以决策的大型设计调整,更不应该需要单独立项来推进。这些常规变更应该可以融入每日或者每周的日常系统维护中去完成。
我们怎么能够预知某个系统未来的变更需求,以便提前做准备呢?我们怎么能在没有水晶球与时光穿梭机的情况下,未卜先知,降低未来的变更成本呢?
所谓软件架构,就是你希望在项目一开始就能做对,但是却不一定能够做得对的决策的集合。
——Ralph Johnson
了解历史已经够难了,我们对现实的认知也不够可靠,预言未来就更难了。
这就是不同的软件开发理论的主要分歧点。
其中一条比较悲观阴暗的路线认为,只有权威和刚性才能带来强壮与稳定。如果某项变更成本高昂,那么就应该忽视它——变更背后的需求要么应该被抑制,要么就应该被丢到官僚主义的大机器中去绞碎。架构师的决定永远是完整的、彻底的,软件架构就是全体开发人员的敌托邦噩梦(Dystopia),永远是所有人沮丧的源泉。
另外一条路线则到处充斥着大量的投机性的通用设计。在这样的软件项目中到处都是硬编码的猜测性代码,到处是无穷无尽的参数,存在着成篇累牍的无效代码。 维护这样的项目,肯定会遇到意外情况,而且无论预留多少资源都不够应付。
而本书试图探索的则是一条整洁路线。这条路线拥抱软件的灵活多变性,将其作为系统的一级设计目标。同时,我们也承认人类并不能全知全晓,但在信息不全的情况下人类仍然能够做出优良的决策。这条路线可以让我们多发挥优势,避开弱势。通过实际创造和探索,不停地提出问题和进行实验。优良的软件架构不是一成不变的,只有经过不断打磨和改进才能最终成就。
软件架构是一个猜想,只有通过实际实现和测量才能证实。
——Tom Gilb
遵循这条路线,我们需要用心,全神贯注,不停观察和思考,在原则指导下不断实践。虽然这可能听起来很麻烦、很慢,但是只要坚持走下去一定能够成功。
走快的唯一方法是先走好。
——Robert C. Martin
一起享受这个过程吧!
Kevlin Henney
2017年5月
前言
本书的名字叫作《架构整洁之道》,使用这个名字可谓是十分胆大,甚至可以说有点目中无人了。那么,为什么我会选择写这本书,并且使用这个名字呢?
自1964年,12岁的我写下了人生的第一行代码算起,到2016年,我已经编程超过50年。在这段时间里,我自认为学到了构建软件系统的一些方法——并且我相信这些方法和经验对其他人应该有些价值。
我学习的途径是实际构建一些大大小小的软件系统。我写过小型的嵌入式系统,也构造过大型的批处理系统;我构建过实时控制系统,也构建过Web网页系统;我写过命令行程序、图形界面程序、进程管理程序、游戏、计费系统、通信系统、设计工具、画图工具等。
我写过单线程程序,也写过多线程程序;我写过由几个重型进程组成的应用,也写过由大量轻型进程组成的应用;我写过跨多个处理器的应用,还有数据库类、数值计算类和几何计算类应用,以及很多很多其他类型的应用。
回首过去,经历了这么多应用和系统的构建过程,我最意外的领悟是:
软件架构的规则是相同的!
我所构建的这些系统是千差万别的,为什么所有这些差异巨大的系统都遵守同样的软件架构规则呢?这里我得出的结论是,软件架构规则和其他变量完全无关。
回顾过去这半个世纪以来硬件系统产生的巨大变革,这个结论就更惊人了。我的编程生涯起步于像家用冰箱那么大的巨型机时代,它的CPU频率只有0.5MHz,拥有4KB核心内存,32KB磁盘存储,以及每秒只能传输10个字符的电传打字机接口。而现在,我正在一辆游览南非的观光车上敲这篇前言。我正在用一个拥有4核i7的MacBook,每核2.8GHz。这台笔记本电脑有16GB内存,1TB SSD硬盘,可以用2880?1800虹膜显示屏展现高清视频。二者计算能力上的差距真的是天壤之别。粗略分析可知,这台MacBook至少比我半个世纪以前用的计算机强大1022倍。
22个数量级的差距是非常非常巨大的,从地球到半人马星系也只有1022埃(angstrom,长度单位,主要用来描述原子尺寸与波长),你口袋里的零钱加起来所包含的电子数量也差不多为1022个。而这个数字(注意还是至少)是我在一生中,所亲身经历的计算能力的提升。
计算能力发生了这么巨大的变化,但对我所写软件的影响有多大呢?软件尺寸当然变大了。我过去认为2000行的程序就很庞大了。毕竟这样的程序变成打孔卡片能装满一盒子,重量超过10磅。而现在,一个10万行的程序都不能算大程序了。
同时,软件性能当然也有大幅提升。我们现在可以轻轻松松地完成那些1960年只能幻想的事情。电影The Forbin Project、The Moon is a Harsh Mistress以及2001: A Space Odyssey都试图预言我们的现状,但是都没有成功。在这些电影中普遍展现的是获得了智能的巨型机器,而我们目前所拥有的计算机,虽然体积之小是当初难以想象的,却还仅仅只是机器。
同时,还有一点很重要,今天的软件与过去的软件本质上仍然是一样的。都是由if语句、赋值语句以及while循环组成的。
哦,你可能会抗议说我们现在有更好的编程语以及更先进的编程范式了。毕竟,我们现在都是用Java、C#、Ruby语言编写程序,并且大量采用面向对象编程方式。这没错,但是最终产生的代码仍然只是顺序结构、分支结构、循环结构的组合,这方面和20世纪60年代甚至50年代的程序是一模一样的。
如果深入研究计算机编程的本质,我们就会发现这50年来,计算机编程基本没有什么大的变化。编程语言稍微进步了一点,工具的质量大大提升了,但是计算机程序的基本构造没有什么变化。
如果一个1966年的计算机程序员时空穿梭来到2016年,在我的MacBook上用IntelliJ写Java程序,她 可能也就需要24小时来适应一下,然后很快就能照常工作了。Java其实和C区别并不大,和FORTRAN也没那么大区别。
同样,如果我把你——读者传送回1966年,告诉你如何在一个每秒处理10个字符的终端上通过打孔纸带来编辑PDP-8代码,估计你最多也只需要24小时的适应时间。毕竟编程还是编程,代码并没有本质的变化。
这就是秘密所在:计算机代码没有变化,软件架构的规则也就一直保持了一致。软件架构的规则其实就是排列组合代码块的规则。由于这些代码块本质上没有变化,因此排列组合它们的规则也就不会变化。
年轻的一代程序员可能认为这些都是胡说。他们可能坚持认为现在所有东西都是崭新的、从来没有过的,过去的规则已经过时,不再适用了。这是一个非常大的错误。这些规则一直都没有变。虽然我们有了新的编程语言、新的编程框架、新的编程范式,但是软件架构的规则仍然和1946年阿兰·图灵写下第一行机器代码的时候一样。
当然,不一样的是,那时候我们还不知道规则是什么。所以我们一次一次地颠覆了它,并且为此一次一次地付出了代价。半个世纪过去了,我们终于可以说,现在我们对这些规则有一定程度的了解了。
写这本书就是为了讲述这些规则,这些永恒的、不变的软件架构规则。
第1部分 概述
第1章 设计与架构究竟是什么 3
目标是什么 4
案例分析 5
本章小结 11
第2章 两个价值维度 12
行为价值 13
架构价值 13
哪个价值维度更重要 14
艾森豪威尔矩阵 15
为好的软件架构而持续斗争 16
第2部分 从基础构件开始:编程范式
第3章 编程范式总览 21
结构化编程 22
面向对象编程 22
函数式编程 23
仅供思考 23
本章小结 24
第4章 结构化编程 25
可推导性 26
goto是有害的 28
功能性降解拆分 29
形式化证明没有发生 29
科学来救场 29
测试 30
本章小结 31
第5章 面向对象编程 32
封装 33
继承 36
多态 38
本章小结 44
第6章 函数式编程 45
整数平方 46
不可变性与软件架构 47
可变性的隔离 48
事件溯源 49
本章小结 51
第3部分 设计原则
第7章 SRP:单一职责原则 56
反面案例2:代码合并 59
解决方案 60
本章小结 61
第8章 OCP:开闭原则 62
思想实验 63
依赖方向的控制 67
信息隐藏 67
本章小结 67
第9章 LSP:里氏替换原则 68
继承的使用指导 69
正方形/长方形问题 70
LSP与软件架构 70
违反LSP的案例 71
本章小结 73
第10章 ISP:接口隔离原则 74
ISP与编程语言 76
ISP与软件架构 76
本章小结 77
第11章 DIP:依赖反转原则 78
稳定的抽象层 79
工厂模式 80
具体实现组件 82
本章小结 82
第4部分 组件构建原则
第12章 组件 84
组件发展史 85
重定位技术 88
链接器 88
本章小结 90
第13章 组件聚合 91
复用/发布等同原则 92
共同闭包原则 93
共同复用原则 94
组件聚合张力图 95
本章小结 97
第14章 组件耦合 98
无依赖环原则 99
自上而下的设计 105
稳定依赖原则 106
稳定抽象原则 112
本章小结 117
第5部分 软件架构
第15章 什么是软件架构 120
开发(Development) 122
部署(Deployment) 123
运行(Operation) 123
维护(Maintenance) 124
保持可选项 124
设备无关性 126
垃圾邮件 128
物理地址寻址 129
本章小结 130
第16章 独立性 131
用例 132
运行 133
开发 133
部署 134
保留可选项 134
按层解耦 135
用例的解耦 136
解耦的模式 136
开发的独立性 137
部署的独立性 137
重复 138
再谈解耦模式 139
本章小结 141
第17章 划分边界 142
几个悲伤的故事 143
FitNesse 146
应在何时、何处画这些线 148
输入和输出怎么办 151
插件式架构 152
插件式架构的好处 153
本章小结 154
第18章 边界剖析 155
跨边界调用 156
令人生畏的单体结构 156
部署层次的组件 158
线程 159
本地进程 159
服务 160
本章小结 161
第19章 策略与层次 162
层次(Level) 163
本章小结 166
第20章 业务逻辑 167
业务实体 168
用例 169
请求和响应模型 171
本章小结 172
第21章 尖叫的软件架构 173
架构设计的主题 174
架构设计的核心目标 175
那Web呢 175
框架是工具而不是生活信条 175
可测试的架构设计 176
本章小结 176
第22章 整洁架构 177
依赖关系规则 179
一个常见的应用场景 183
本章小结 184
第23章 展示器和谦卑对象 185
谦卑对象模式 186
展示器与视图 186
测试与架构 187
数据库网关 188
数据映射器 188
服务监听器 189
本章小结 189
第24章 不完全边界 190
省掉最后一步 191
单向边界 192
门户模式 193
本章小结 193
第25章 层次与边界 194
基于文本的冒险游戏:Hunt The Wumpus 195
可否采用整洁架构 196
交汇数据流 199
数据流的分割 199
本章小结 201
第26章 Main组件 203
最细节化的部分 204
本章小结 208
第27章 服务:宏观与微观 209
面向服务的架构 210
服务所带来的好处 210
运送猫咪的难题 212
对象化是救星 213
基于组件的服务 215
横跨型变更 216
本章小结 216
第28章 测试边界 217
测试也是一种系统组件 218
可测试性设计 219
测试专用API 220
本章小结 221
第29章 整洁的嵌入式架构 222
“程序适用测试”测试 225
目标硬件瓶颈 228
本章小结 238
第6部分 实现细节
第30章 数据库只是实现细节 240
关系型数据库 241
为什么数据库系统如此流行 242
假设磁盘不存在会怎样 243
实现细节 243
但性能怎么办呢 244
一段轶事 244
本章小结 246
第31章 Web是实现细节 247
无尽的钟摆 248
总结一下 250
本章小结 251
第32章 应用程序框架是实现细节 252
框架作者 253
单向婚姻 253
风险 254
解决方案 255
不得不接受的依赖 255
本章小结 256
第33章 案例分析:视频销售网站 257
产品 258
用例分析 258
组件架构 260
依赖关系管理 261
本章小结 262
第34章 拾遗 263
按层封装 264
按功能封装 266
端口和适配器 268
按组件封装 270
具体实现细节中的陷阱 274
组织形式与封装的区别 275
其他的解耦合模式 277
本章小结:本书拾遗 279
后序 280
附录A 架构设计考古 283