自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Go 中常用的四大重構技術

開發(fā) 后端
這是我常應用于代碼的技術。它包括提取一段按意圖分組的代碼,并轉(zhuǎn)移到新方法中。通過提取可以將一個長方法或函數(shù)拆分為一些小方法,這些小方法將邏輯組合在一起。通常,小方法或函數(shù)的名稱可以更好地了解該邏輯是什么。

[[433087]]

大家好,我是程序員幽鬼。

Martin Fowler 在他的書中[1]將重構定義為*“對軟件的內(nèi)部結構進行的更改,以使其更易于理解,并且在不更改其可觀察到的行為的情況下更低廉地進行修改”*。本書包含大量重構技術,這些重構技術旨在在某些情況下應用,并旨在消除代碼壞味道[2]。

重構是一個非常廣泛的話題,我發(fā)現(xiàn)重構在軟件開發(fā)過程中起著重要的作用。它們的相關性很高,因此它們是TDD[3]生命周期的重要組成部分。

由于它們的重要性,在這篇文章中,我想分享一下軟件開發(fā)人員中使用最多的 4 種重構技術。但是在開始之前,因為可以自動應用重構技術(即某些 IDE 為你提供了幫助,通過應用重構工具,只需單擊幾下鼠標和進行選擇,即可使你的生活更輕松),在這里,我將通過使用 Go 語言手動重構進行描述,并嘗試將其作為參考指南。我們的開發(fā)團隊意識到,在應用任何重構技術之前,應將可觀察到的功能包含在單元測試中,并通過所有測試。

01 提取方法

這是我常應用于代碼的技術。它包括提取一段按意圖分組的代碼,并轉(zhuǎn)移到新方法中。通過提取可以將一個長方法或函數(shù)拆分為一些小方法,這些小方法將邏輯組合在一起。通常,小方法或函數(shù)的名稱可以更好地了解該邏輯是什么。

下面的示例顯示了應用此重構技術之前和之后的情況。我的主要目標是通過將復雜度分為不同的功能,這樣來抽象其復雜度。

  1. func StringCalculator(exp string) int { 
  2.     if exp == "" { 
  3.         return 0 
  4.     } 
  5.      
  6.     var sum int 
  7.     for _, number := range strings.Split(exp, ",") { 
  8.         n, err := strconv.Atoi(number) 
  9.         if err != nil { 
  10.             return 0 
  11.         } 
  12.         sum += n 
  13.     } 
  14.     return sum 

重構為:

  1. func StringCalculator(exp string) int { 
  2.     if exp == "" { 
  3.         return 0 
  4.     } 
  5.  return sumAllNumberInExpression(exp) 
  6.  
  7. func sumAllNumberInExpression(exp string) int { 
  8.     var sum int 
  9.     for _, number := range strings.Split(exp, ",") { 
  10.         sum += toInt(number) 
  11.     } 
  12.     return sum 
  13.  
  14. func toInt(exp string) int { 
  15.     n, err := strconv.Atoi(exp) 
  16.     if err != nil { 
  17.         return 0 
  18.     } 
  19.     return n 

StringCalculator 函數(shù)更簡單了,但是當添加了兩個新的函數(shù)時,它會增加復雜性。這是一個我愿意做出慎重決定的犧牲,我將此作為參考而不是規(guī)則,從某種意義上說,了解應用重構技術的結果可以很好地判斷是否應用重構技術。

02 移動方法

有時,在使用提取方法后,我發(fā)現(xiàn)了另一個問題:此方法應該屬于此結構或包嗎?Move Method 是一種簡單的技術,包括將方法從一個結構移動到另一個結構。我發(fā)現(xiàn)一個技巧,來確定某個方法是否應該屬于該結構:弄清楚該方法是否訪問了另一個結構依賴項的內(nèi)部??聪旅娴睦樱?/p>

  1. type Book struct { 
  2.     ID    int 
  3.     Title string 
  4.  
  5. type Books []Book 
  6.  
  7. type User struct { 
  8.     ID    int 
  9.     Name  string 
  10.     Books Books 
  11.  
  12. func (u User) Info() { 
  13.     fmt.Printf("ID:%d - Name:%s", u.ID, u.Name
  14.     fmt.Printf("Books:%d", len(u.Books)) 
  15.     fmt.Printf("Books titles: %s", u.BooksTitles()) 
  16.  
  17. func (u User) BooksTitles() string { 
  18.     var titles []string 
  19.     for _, book := range u.Books { 
  20.         titles = append(titles, book.Title) 
  21.     } 
  22.     return strings.Join(titles, ","

如你所見,User 的方法BooksTitles 使用了 books(具體是 Title)中的內(nèi)部字段多于User,這表明該方法應歸于Books。應用這種重構技術將該方法移動到Books類型上,然后由用戶的Info方法調(diào)用。

  1. func (b Books) Titles() string { 
  2.     var titles []string 
  3.     for _, book := range b { 
  4.         titles = append(titles, book.Title) 
  5.     } 
  6.     return strings.Join(titles, ","
  7.  
  8. func (u User) Info() { 
  9.     fmt.Printf("ID:%d - Name:%s", u.ID, u.Name
  10.     fmt.Printf("Books:%d", len(u.Books)) 
  11.     fmt.Printf("Books titles: %s", u.Books.Titles()) 

應用此方法后,Books類型會更內(nèi)聚,因為它是唯一擁有控制權和對它的字段和內(nèi)部屬性訪問權的人。同樣,這是在深思熟慮之前進行的思考過程,知道應用重構會帶來什么結果。

03 引入?yún)?shù)對象

你見過多少像下面方法一樣,有很多參數(shù)的:

  1. func (om *OrderManager) Filter(startDate, endDate time.Time, country, state, city, status string) (Orders, error) { 
  2.     ... 

即使我們看不到函數(shù)內(nèi)部的代碼,當我們看到大量這樣的參數(shù)時,我們也可以考慮它執(zhí)行的大量操作。

有時,我發(fā)現(xiàn)這些參數(shù)之間高度相關,并在以后定義它們的方法中一起使用。這為重構提供了一種使該場景更加面向?qū)ο蟮姆绞竭M行處理的方法,并且建議將這些參數(shù)分組為一個結構,替換方法簽名以將該對象用作參數(shù),并在方法內(nèi)部使用該對象。

  1. type OrderFilter struct { 
  2.     StartDate time.Time 
  3.     EndDate   time.Time 
  4.     Country   string 
  5.     State     string 
  6.     City      string 
  7.     Status    string 
  8.  
  9. func (om *OrderManager) Filter(of OrderFilter) (Orders, error) { 
  10.     // use of.StartDate, of.EndDate, of.Country, of.State, of.City, of.Status. 

看起來更干凈,并且可以確定這些參數(shù)的身份,但是這將要求我更改調(diào)用此方法的所有引用,并且需要OrderFilter在傳遞給該方法之前創(chuàng)建一個新類型的對象作為參數(shù)。同樣,在嘗試進行此重構之前,我會盡力思考并考慮后果。當你的代碼中的影響程度很低時,我認為此技術非常有效。

04 用符號常量替換魔數(shù)

該技術包括用常數(shù)變量替換硬編碼值以賦予其意圖和意義。

  1. func Add(input string) int { 
  2.     if input == "" { 
  3.         return 0 
  4.     } 
  5.  
  6.     if strings.Contains(input, ";") { 
  7.         n1 := toNumber(input[:strings.Index(input, ";")]) 
  8.         n2 := toNumber(input[strings.Index(input, ";")+1:]) 
  9.  
  10.         return n1 + n2 
  11.     } 
  12.  
  13.     return toNumber(input) 
  14.  
  15. func toNumber(input string) int { 
  16.     n, err := strconv.Atoi(input) 
  17.     if err != nil { 
  18.         return 0 
  19.     } 
  20.     return n 

其中 ; 字符是什么意思?如果答案對我來說不太明確,我可以創(chuàng)建一個臨時變量,并使用硬編碼字符設置該值,以賦予其意義。

  1. func Add(input string) int { 
  2.     if input == "" { 
  3.         return 0 
  4.     } 
  5.  
  6.     numberSeparator := ";" 
  7.     if strings.Contains(input, numberSeparator) { 
  8.         n1 := toNumber(input[:strings.Index(input, numberSeparator)]) 
  9.         n2 := toNumber(input[strings.Index(input, numberSeparator)+1:]) 
  10.  
  11.         return n1 + n2 
  12.     } 
  13.  
  14.     return toNumber(input) 
  15.  
  16.  
  17. func toNumber(input string) int { 
  18.     n, err := strconv.Atoi(input) 
  19.     if err != nil { 
  20.         return 0 
  21.     } 
  22.     return n 

總結

感謝閱讀,希望對你有所幫助。重構是一個非常廣泛的話題,本文舉例說明了重構中使用最多的四個。不要將此處提到的內(nèi)容視為理所當然,自己嘗試一下。此處描述的重構技術僅用作指導原則,而未作為規(guī)則遵循,意味著它們在需要時可以有針對性地進行調(diào)整。最后,我想說我們對所編寫的所有代碼和所使用的所有工具負責,我們的經(jīng)驗和知識可以指導我們掌握在每種情況下最適合的技能,我認為重構技術確實值得。

原文鏈接:https://wawand.co/blog/posts/four-most-refactoring-techniques-i-use/

參考資料

[1]書中: https://martinfowler.com/books/refactoring.html

[2]壞味道代碼: https://en.wikipedia.org/wiki/Code_smell

[3]TDD: https://en.wikipedia.org/wiki/Test-driven_development#/media/File:TDD_Global_Lifecycle.png

本文轉(zhuǎn)載自微信公眾號「幽鬼」,可以通過以下二維碼關注。轉(zhuǎn)載本文請聯(lián)系幽鬼公眾號。

 

責任編輯:武曉燕 來源: 幽鬼
相關推薦

2015-04-02 15:50:47

數(shù)據(jù)倉庫數(shù)據(jù)挖掘

2021-09-17 10:50:14

容器 Linux

2010-09-15 13:35:25

SwingHibernateStruts

2020-02-29 09:09:34

物聯(lián)網(wǎng)定位技術IOT

2020-04-06 20:11:26

區(qū)塊鏈分布式核心技術

2020-06-29 14:52:31

隱私增強技術PET加密

2011-03-21 09:01:49

CSS框架

2015-07-17 09:50:16

Carthage優(yōu)劣比較

2017-03-17 19:48:01

人臉識別

2015-12-04 13:46:31

技術人創(chuàng)業(yè)創(chuàng)業(yè)項目

2024-07-15 08:13:12

BERT模型嵌入

2011-06-22 08:38:35

Java EE

2014-03-27 15:57:45

Android組件Activity

2013-09-02 10:17:05

虛擬化技術

2012-07-11 10:41:38

虛擬化

2020-09-22 15:22:04

物聯(lián)網(wǎng)智慧城市技術

2012-05-29 09:23:23

HTML5移動應用框架jQTouch

2020-07-28 08:48:49

Python繪圖工具

2013-01-06 10:44:43

微軟Windows 8云計算

2016-03-30 11:51:55

點贊
收藏

51CTO技術棧公眾號