ch10 I Can't Run This Method in a Test Harness
要為一段 code 寫測試,首先要在測試中建立它所屬的 class 的 instance,接著為要修改的 method 寫測試。這章要解決的是「難以在測試中執行要測試的 method」的情境。為一個 method 寫測試可能會遇到的問題:
- 無法在測試中 access 那個 method,例如它可能是 private 的或是有其他存取限制。
- 很難建立所需參數,導致很難 call method,例如參數是一包 XML。
- 要測試的 method 可能產生糟糕的 side effect,例如修改 DB、發射飛彈等等,所以無法在測試中執行它。
- 需要透過該 method 使用的 object 進行 sense,才能知道這個 method 做了什麼事。
The Case of the Hidden Method
假設我們要修改的是一個 private method,想要測試它,該怎麼辦呢?
首先,能透過 public method 來測試它嗎?如果可以,就這麼幹吧~用 public method 去測試,就是按照程式中 private method 如何被使用的方式去測試它。如果有天要把 private method 改成 public,把它變成 public 的人應該寫一系列的測試說明這個 method 的用途以及 caller 該如何使用它。
這邊有提到一點 method 設計實作上的小概念:
雖然 general 的 method 對 caller 來說蠻有用的,但每個 method 的功能應該剛好可以滿足 caller 並且易於理解與修改。
有時候呢,我們就是想直接為 private method 寫測試(任性),可能是因為我們想用測試來知道如何使用 private method,或者用 public method 來測試它實在太難太痛苦啦~
例如一個擁有商業邏輯並且會 call third-party API 的 class 做的事情是:call API 取得一包 XML 資料,parse XML 得到商業邏輯需要的資料,再做商業邏輯上的計算或操作。我們想知道 parse XML 的 private method 是否正確,但它埋在整個流程裡,而用 public method 做整個的 call API、parsing、商業邏輯的測試難以只測試到 parse XML 的部份。
所以,想為 private method 寫測試時該怎麼辦呢?
如果需要測試一個 private method,就該把它設成 public。
看到書上這句話我蠻驚恐的,想著:「等等等,不是吧?就這樣直接把 private method 變成 public 好嗎?這不會在 class 上開出看起來突兀或者不知如何使用的 method 嗎?」
如果不方便將其設為 public,大多數情況下意味著我們的 class 做太多事了,應該進行調整。
喔~原來是這樣~這倒是真的~像上面那個例子,一個 class 既 call third-party API 又 parse XML 又做商業邏輯,太多事情了。
如果我們想測試一個 private method,首先看它是不是個適合在這個 class 當作 public 的 method?如果是,直接改成 public。否則看看是不是這個 class 做太多事了,有些事可以交由另一個 class 處理。例如我們把 parse XML 有關的 method 放到另一個 parser class,這些 method 到了新 class 會變成 public,原本的 class 就能 new 一個 parser 出來做事。
好的設計應當是可測試的,不具可測試性的設計是糟糕的。
如果我們遇到上面這樣的狀況,想拆解 class、將職責分開,卻沒有多少現成測試呢?假設我們想拆解這個包山包海的 class,但它本身卻沒有什麼測試,而 refactor 應當要有測試保護,雞生蛋蛋生雞的問題出現啦~
又或許,我們正在開發週期的後期,軟體已經接近 deploy,我們沒有多少時間去做拆解的 refactor,而且沒有測試又讓 refactor 的風險大幅提昇。儘管 refactor 可以改善程式結構、有好處,但需要考慮目前處於開發週期的哪個階段、時間多寡與風險高低,才能決定要不要 refactor 以及 refactor 到什麼程度。
時間不夠或風險太高的時,我們無法拆解 class。退而求其次,至少幫我們要修改的 private method 加上測試,讓這個 class 開始有測試保護也是好的。
想為 private method 加上測試,表示要能在測試中直接 call 到這個 method,要怎麼做呢?
又用到 Extract and Override 了~這招也太萬用…
假設我們有個與機票搜尋、訂購有關的 class:
1 | class Flight { |
在 search()
裡先用 http client 向 third party API 送 request 並收 response,response 是一包 XML,我們想 parse 出其中需要的資料而 call parseSearchResponse()
。因為 XML 相當複雜,我們希望能單獨確認 parsing 結果是否正確。
這個 class 除了 search、booking、parse 各種 XML 之外還會做許多事情,它的職責太多了,如果我們現在沒有時間去拆解它,卻想測試 parseSearchResponse()
的結果該怎麼做?
首先將 parseSearchResponse()
從 private
變成 protected
:
1 | class Flight { |
接著在測試中繼承它:
1 | class FlightForTest extends Flight { |
這樣就能在測試 call 到 parseSearchResponse()
進行測試了:
1 | class FlightTest extends TestCase { |
這麼做雖然沒有立即改善 Flight
做太多事的問題,但至少幫修改的地方加上測試,確保目前的修改是正確的。並且為將來拆解 class 的 refactor 鋪路――因為加了些測試而減少之後 refactor 所需的 effort 跟時間,使之後 refactor 成為可能。