大話重構連載14:我們是這樣自動化測試的


說了那么多,讓我們用示例看看,系統重構是應該怎樣做自動化測試的。還是回到前面那個HelloWorld的例子(詳見 3.3 小步快跑是這樣玩的),該類中有一個sayHello()方法,只要我們輸入當前的時間與用戶名,就返回對該用戶的問候語。如果當前時間是上午,則返回“Hi, XXX. Good morning!”;如果是下午,則返回“Hi, XXX. Good afternoon!”;如果是晚上,則返回“Hi, XXX. Good Night!”,這是HelloWorld這個程序實現的功能。

然后我們開始為這段程序編寫測試代碼(如果采用測試驅動開發,應當先寫測試代碼再寫程序)。我們首先建立一個test源程序目錄,然后建立與被測程序對應的包和測試程序。這就是說,如果被測程序在“org.refactoring.helloWorld.resource”包中,則測試程序應當建立“test.org.refactoring.helloWorld.resource”包與之對應;如果被測程序叫“HelloWorld”,則建立“HelloWorldTest”類與之對應,這個類是一個JUnit測試程序。

下面就是編寫這個測試程序執行測試了。由於被測程序有三個分支,即當前時間是上午、下午、晚上,因此我們分別為之建立了三個測試用例,測試程序如下:

 1 /**
 2  * Test for {@link org.refactoring.helloWorld.resource.HelloWorld}
 3  * @author fangang
 4  */
 5 public class HelloWorldTest {
 6 
 7     private HelloWorld helloWorld = null;
 8     /**
 9      * @throws java.lang.Exception
10      */
11     @Before
12     public void setUp() throws Exception {
13         helloWorld = new HelloWorld();
14     }
15 
16     /**
17      * @throws java.lang.Exception
18      */
19     @After
20     public void tearDown() throws Exception {
21         helloWorld = null;
22     }
23 
24     /**
25      * Test method for {@link org...HelloWorld#sayHello(java.util.Date, java.lang.String)}.
26      */
27     @Test
28     public void testSayHelloInTheMorning() {
29         Date now = DateUtil.createDate(2013, 9, 7, 9, 23, 11);
30         String user = "鮑曉妹";
31         String result = "";
32         result = helloWorld.sayHello(now, user);
33         assertThat(result, is("Hi, 鮑曉妹. Good morning!"));
34     }
35 
36     /**
37      * Test method for {@link org...HelloWorld#sayHello(java.util.Date, java.lang.String)}.
38      */
39     @Test
40     public void testSayHelloInTheAfternoon() {
41         Date now = DateUtil.createDate(2013, 9, 7, 15, 7, 10);
42         String user = "關二鍋";
43         String result = "";
44         result = helloWorld.sayHello(now, user);
45         assertThat(result, is("Hi, 關二鍋. Good afternoon!"));
46     }
47 
48     /**
49      * Test method for {@link org...HelloWorld#sayHello(java.util.Date, java.lang.String)}.
50      */
51     @Test
52     public void testSayHelloAtNight() {
53         Date now = DateUtil.createDate(2013, 9, 7, 21, 30, 10);
54         String user = "IT攻城獅";
55         String result = "";
56         result = helloWorld.sayHello(now, user);
57         assertThat(result, is("Hi, IT攻城獅. Good night!"));
58     }
59 }

 

這段程序采用的是JUnit4編寫的,其中assertThat(result, is("Hi, IT攻城獅. Good night!"));,第一個參數是被測程序執行的結果,而第二個參數是根據期望結果進行驗證。如果執行結果與預期結果相同,則測試通過,否則測試失敗。

隨后我們運行該測試程序,得到如下結果:

 

圖4.1 JUnit測試結果

 

三項測試用例全部通過,測試成功!

現在我們為原程序編寫了測試用例並全部測試通過,我們為重構所做的准備工作就一切就緒了。然后,我們開始進行第一次重構。如前面所述,第一次重構我們調整了程序的順序,進行了分段,增加了注釋,並修改了相應的變量,使其更加利於閱讀。這是一個小步快跑的過程,我們完成此次重構只花費了3、5分鍾。當重構完成,程序重新回到可編譯運行狀態時,我們執行它的這個測試程序,測試通過。測試通過意味着,雖然程序內部的代碼有所修改,但程序對外的功能沒有變化,即程序的外部行為沒有變化,則重構成功,我們可以繼續后面的工作。

第二次重構,我們運用“抽取方法”,從sayHello()函數中抽取出了getFirstGreeting(), getSecondGreeting(), getHour()三個方法。之后我們再次執行測試程序,測試通過。

第三次重構,我們運用“抽取類”,將getFirstGreeting()與getSecondGreeting()分別抽取出來形成了GreetingToUser和GreetingAboutTime。完成之后執行測試通過。

第四次重構,我們的需求發生了變化,問候語不僅隨一天中的上午、下午、晚上等進行變化,還需要根據不同的日期判斷是否是節日。在這種情況下,我們采用“兩頂帽子”的方式進行開發:首先不引入新的需求,僅僅修改原程序,使之適應新需求。為此我們從GreetingAboutTime類中提煉出DateUtil,使之不僅有getHour(),還有getMonth()與getDate()。完成重構以后測試通過。

關於“兩頂帽子”的設計方式,也是系統重構中另一個不同以往的地方,我們還將在后面詳細地進行討論。隨后我們開始添加新需求,使GreetingAboutTime中的getGreeting()寫成這樣:

 1 /**
 2  * Test for {@link org.refactoring.helloWorld.resource.HelloWorld}
 3  * @author fangang
 4  */
 5 public class HelloWorldTest {
 6 
 7     private HelloWorld helloWorld = null;
 8     /**
 9      * @throws java.lang.Exception
10      */
11     @Before
12     public void setUp() throws Exception {
13         helloWorld = new HelloWorld();
14     }
15 
16     /**
17      * @throws java.lang.Exception
18      */
19     @After
20     public void tearDown() throws Exception {
21         helloWorld = null;
22     }
23 
24     /**
25      * Test method for {@link org...HelloWorld#sayHello(java.util.Date, java.lang.String)}.
26      */
27     @Test
28     public void testSayHelloInTheMorning() {
29         Date now = DateUtil.createDate(2013, 9, 7, 9, 23, 11);
30         String user = "鮑曉妹";
31         String result = "";
32         result = helloWorld.sayHello(now, user);
33         assertThat(result, is("Hi, 鮑曉妹. Good morning!"));
34     }
35 
36     /**
37      * Test method for {@link org...HelloWorld#sayHello(java.util.Date, java.lang.String)}.
38      */
39     @Test
40     public void testSayHelloInTheAfternoon() {
41         Date now = DateUtil.createDate(2013, 9, 7, 15, 7, 10);
42         String user = "關二鍋";
43         String result = "";
44         result = helloWorld.sayHello(now, user);
45         assertThat(result, is("Hi, 關二鍋. Good afternoon!"));
46     }
47 
48     /**
49      * Test method for {@link org...HelloWorld#sayHello(java.util.Date, java.lang.String)}.
50      */
51     @Test
52     public void testSayHelloAtNight() {
53         Date now = DateUtil.createDate(2013, 9, 7, 21, 30, 10);
54         String user = "IT攻城獅";
55         String result = "";
56         result = helloWorld.sayHello(now, user);
57         assertThat(result, is("Hi, IT攻城獅. Good night!"));
58     }
59 }

 

之后我們的測試不能通過:

 

圖4.2 測試用例不能通過

 

為什么testSayHelloAtNight測試不能通過呢?仔細查看被測程序,我們發現它的功能發生了變化,變為:如果當前時間是1月1日,則返回“Hi, XXX. Happy new year!”;如果是1月14日,則返回“Hi, XXX. Happy valentine's day!”……如果當前時間都不是這些節日,如果是上午則返回“Hi, XXX. Good morning!”,是中午則返回“Hi, XXX. Good noon!”,是下午則返回“Hi, XXX. Good afternoon!”,是傍晚則返回“Hi, XXX. Good evening!”,否則才返回“Hi, XXX. Good night!”。正因為如此,我們需要調整我們的測試程序,為每一個分支編寫測試用例。測試修改好后,最后測試通過。

大話重構連載首頁:http://www.cnblogs.com/mooodo/p/talkAboutRefactoringHome.html

特別說明:希望網友們在轉載本文時,應當注明作者或出處,以示對作者的尊重,謝謝!


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM