重構 改善既有代碼的設計pdf

2019年6月28日22:38:47 評論 58
摘要

軟件開發的不朽經典
生動闡述重構原理和具體做法
普通程序員進階到編程高手必須修煉的秘笈
《重構 改善既有代碼的設計》凝聚了軟件開發社區專家多年摸索而獲得的寶貴經驗,擁有不因時光流逝而磨滅的價值。今天,無論是重構本身,業界對重構的理解,還是開發工具對重構的支持力度,都與本書出版時不可同日而語,但書中所蘊涵的意味和精華,依然值得反復咀嚼,而且往往能夠常讀常新。

重構 改善既有代碼的設計 內容簡介

《重構 改善既有代碼的設計》清晰揭示了重構的過程,解釋了重構的原理和實踐方式,并給出了何時以及何地應該開始挖掘代碼以求改善。書中給出了70 多個可行的重構,每個重構都介紹了一種經過驗證的代碼變換手法的動機和技術。《重構 改善既有代碼的設計》提出的重構準則將幫助你一次一小步地修改你的代碼,從而減少了開發過程中的風險。

重構 改善既有代碼的設計 目錄

第1章 重構,第一個案例 1

1.1 起點 1

1.2 重構的第一步 7

1.3 分解并重組statement() 8

1.4 運用多態取代與價格相關的條件邏輯 34

1.5 結語 52

第2章 重構原則 53

2.1 何謂重構 53

2.2 為何重構 55

2.3 何時重構 57

2.4 怎么對經理說 60

2.5 重構的難題 62

2.6 重構與設計 66

2.7 重構與性能 69

2.8 重構起源何處 71

第3章 代碼的壞味道 75

3.1 Duplicated Code(重復代碼) 76

3.2 Long Method(過長函數) 76

3.3 Large Class(過大的類) 78

3.4 Long Parameter List(過長參數列) 78

3.5 Divergent Change(發散式變化) 79

3.6 Shotgun Surgery(霰彈式修改) 80

3.7 Feature Envy(依戀情結) 80

3.8 Data Clumps(數據泥團) 81

3.9 Primitive Obsession(基本類型偏執) 81

3.10 Switch Statements(switch驚悚現身) 82

3.11 Parallel InheritanceHierarchies(平行繼承體系) 83

3.12 Lazy Class(冗贅類) 83

3.13 Speculative Generality(夸夸其談未來性) 83

3.14 Temporary Field(令人迷惑的暫時字段) 84

3.15 Message Chains(過度耦合的消息鏈) 84

3.16 Middle Man(中間人) 85

3.17 Inappropriate Intimacy(狎昵關系) 85

3.18 Alternative Classes with Different Interfaces(異曲同工的類) 85

3.19 Incomplete Library Class(不完美的庫類) 86

3.20 Data Class(純稚的數據類) 86

3.21 Refused Bequest(被拒絕的遺贈) 87

3.22 Comments(過多的注釋) 87

第4章 構筑測試體系 89

4.1 自測試代碼的價值 89

4.2 JUnit測試框架 91

4.3 添加更多測試 97

第5章 重構列表 103

5.1 重構的記錄格式 103

5.2 尋找引用點 105

5.3 這些重構手法有多成熟 106

第6章 重新組織函數 109

6.1 Extract Method(提煉函數) 110

6.2 Inline Method(內聯函數) 117

6.3 Inline Temp(內聯臨時變量) 119

6.4 Replace Temp with Query(以查詢取代臨時變量) 120

6.5 Introduce Explaining Variable(引入解釋性變量) 124

6.6 Split Temporary Variable(分解臨時變量) 128

6.7 Remove Assignments to Parameters(移除對參數的賦值) 131

6.8 Replace Method with Method Object(以函數對象取代函數) 135

6.9 Substitute Algorithm(替換算法) 139

第7章 在對象之間搬移特性 141

7.1 Move Method(搬移函數) 142

7.2 Move Field(搬移字段) 146

7.3 Extract Class(提煉類) 149

7.4 Inline Class(將類內聯化) 154

7.5 Hide Delegate(隱藏“委托關系”) 157

7.6 Remove Middle Man(移除中間人) 160

7.7 Introduce Foreign Method(引入外加函數) 162

7.8 Introduce Local Extension(引入本地擴展) 164

第8章 重新組織數據 169

8.1 Self Encapsulate Field(自封裝字段) 171

8.2 Replace Data Value with Object(以對象取代數據值) 175

8.3 Change Value to Reference(將值對象改為引用對象) 179

8.4 Change Reference to Value(將引用對象改為值對象) 183

8.5 Replace Array with Object(以對象取代數組) 186

8.6 Duplicate Observed Data(復制“被監視數據”) 189

8.7 Change Unidirectional Association to Bidirectional(將單向關聯改為雙向關聯) 197

8.8 Change Bidirectional Association to Unidirectional(將雙向關聯改為單向關聯) 200

8.9 Replace Magic Number with Symbolic Constant(以字面常量取代魔法數) 204

8.10 Encapsulate Field(封裝字段) 206

8.11 Encapsulate Collection(封裝集合) 208

8.12 Replace Record with Data Class(以數據類取代記錄) 217

8.13 Replace Type Code with Class(以類取代類型碼) 218

8.14 Replace Type Code with Subclasses(以子類取代類型碼) 223

8.15 Replace Type Code with State/Strategy(以State/Strategy取代類型碼) 227

8.16 Replace Subclass with Fields(以字段取代子類) 232

第9章 簡化條件表達式 237

9.1 Decompose Conditional(分解條件表達式) 238

9.2 Consolidate Conditional Expression(合并條件表達式) 240

9.3 Consolidate Duplicate Conditional Fragments(合并重復的條件片段) 243

9.4 Remove Control Flag(移除控制標記) 245

9.5 Replace Nested Conditional with Guard Clauses(以衛語句取代嵌套條件表達式) 250

9.6 Replace Conditional with Polymorphism(以多態取代條件表達式) 255

9.7 Introduce Null Object(引入Null對象) 260

9.8 Introduce Assertion(引入斷言) 267

第10章 簡化函數調用 271

10.1 Rename Method(函數改名) 273

10.2 Add Parameter(添加參數) 275

10.3 Remove Parameter(移除參數) 277

10.4 Separate Query from Modifier(將查詢函數和修改函數分離) 279

10.5 Parameterize Method(令函數攜帶參數) 283

10.6 Replace Parameter with Explicit Methods(以明確函數取代參數) 285

10.7 Preserve Whole Object(保持對象完整) 288

10.8 Replace Parameter with Methods(以函數取代參數) 292

10.9 Introduce Parameter Object(引入參數對象) 295

10.10 Remove Setting Method(移除設值函數) 300

10.11 Hide Method(隱藏函數) 303

10.12 Replace Constructor with Factory Method(以工廠函數取代構造函數) 304

10.13 Encapsulate Downcast(封裝向下轉型) 308

10.14 Replace Error Code with Exception(以異常取代錯誤碼) 310

10.15 Replace Exception with Test(以測試取代異常) 315

第11章 處理概括關系 319

11.1 Pull Up Field(字段上移) 320

11.2 Pull Up Method(函數上移) 322

11.3 Pull Up Constructor Body(構造函數本體上移) 325

11.4 Push Down Method(函數下移) 328

11.5 Push Down Field(字段下移) 329

11.6 Extract Subclass(提煉子類) 330

11.7 Extract Superclass(提煉超類) 336

11.8 Extract Interface(提煉接口) 341

11.9 Collapse Hierarchy(折疊繼承體系) 344

11.10 Form Tem Plate Method(塑造模板函數) 345

11.11 Replace Inheritance with Delegation(以委托取代繼承) 352

11.12 Replace Delegation with Inheritance(以繼承取代委托) 355

第12章 大型重構 359

12.1 Tease Apart Inheritance(梳理并分解繼承體系) 362

12.2 Convert Procedural Design to Objects(將過程化設計轉化為對象設計) 368

12.3 Separate Domain from Presentation(將領域和表述/顯示分離) 370

12.4 Extract Hierarchy(提煉繼承體系) 375

第13章 重構,復用與現實 379

13.1 現實的檢驗 380

13.2 為什么開發者不愿意重構他們的程序 381

13.3 再論現實的檢驗 394

13.4 重構的資源和參考資料 394

13.5 從重構聯想到軟件復用和技術傳播 395

13.6 小結 397

13.7 參考文獻 397

第14章 重構工具 401

14.1 使用工具進行重構 401

14.2 重構工具的技術標準 403

14.3 重構工具的實用標準 405

14.4 小結 407

第15章 總結 409

參考書目 413

要點列表 417

索引 419

重構 改善既有代碼的設計 精彩文摘

我該從何說起呢?按照傳統做法,一開始介紹某個東西時,首先應該大致講講它的歷史、主要原理等等。可是每當有人在會場上介紹這些東西,總是誘發我的瞌睡蟲。我的思緒開始游蕩,我的眼神開始迷離,直到主講人秀出實例,我才能夠提起精神。實例之所以可以拯救我于太虛之中,因為它讓我看見事情在真正進行。談原理,很容易流于泛泛,又很難說明如何實際應用。給出一個實例,就可以幫助我把事情認識清楚。

所以我決定從一個實例說起。在此過程中我將告訴你很多重構的道理,并且讓你對重構過程有一點感覺。然后我才能向你展開通常的原理介紹。

但是,面對這個介紹性實例,我遇到了一個大問題。如果我選擇一個大型程序,那么對程序自身的描述和對整個重構過程的描述就太復雜了,任何讀者都不忍卒讀(我試了一下,哪怕稍微復雜一點的例子都會超過100頁)。如果我選擇一個容易理解的小程序,又恐怕看不出重構的價值。

和任何立志要介紹“應用于真實世界中的有用技術”的人一樣,我陷入了一個十分典型的兩難困境。我只能帶引你看看如何在一個我所選擇的小程序中進行重構,然而坦白說,那個程序的規模根本不值得我們那么做。但是如果我給你看的代碼是大系統的一部分,重構技術很快就變得重要起來。所以請你一邊觀賞這個小例子,一邊想象它身處于一個大得多的系統。

實例非常簡單。這是一個影片出租店用的程序,計算每一位顧客的消費金額并打印詳單。操作者告訴程序:顧客租了哪些影片、租期多長,程序便根據租賃時間和影片類型算出費用。影片分為三類:普通片、兒童片和新片。除了計算費用,還要為常客計算積分,積分會根據租片種類是否為新片而有不同。

我用了幾個類來表現這個例子中的元素。圖1-1是一張UML類圖,用以顯示這些類。

我會逐一列出這些類的代碼。

圖書網:重構 改善既有代碼的設計pdf

  • 我的微信
  • 掃一掃加好友
  • weinxin
  • 微信公眾號
  • 掃一掃關注(網站備用地址)
  • weinxin

發表評論

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen: