EasyMock 简介
來源:https://www.ibm.com/developerworks/cn/opensource/os-cn-easymock/
使用注意:
a、靜態方法( static 修飾)無法模擬。
1、使用 EasyMock 進行單元測試
通過 EasyMock,我們可以為指定的接口動態的創建 Mock 對象,并利用 Mock 對象來模擬協同模塊或是領域對象,從而使單元測試順利進行。這個過程大致可以劃分為以下幾個步驟:
?? ???使用 EasyMock 生成 Mock 對象;
?? ???設定 Mock 對象的預期行為和輸出;
?? ???將 Mock 對象切換到 Replay 狀態;
?? ???調用 Mock 對象方法進行單元測試;
?? ???對 Mock 對象的行為進行驗證。
接下來,我們將對以上的幾個步驟逐一進行說明。除了以上的基本步驟外,EasyMock 還對特殊的 Mock 對象類型、特定的參數匹配方式等功能提供了支持,我們將在之后的章節中進行說明。
使用 EasyMock 生成 Mock 對象
根據指定的接口或類,EasyMock 能夠動態的創建 Mock 對象(EasyMock 默認只支持為接口生成 Mock 對象,如果需要為類生成 Mock 對象,在 EasyMock 的主頁上有擴展包可以實現此功能),我們以 ResultSet 接口為例說明EasyMock的功能。java.sql.ResultSet 是每一個 Java 開發人員都非常熟悉的接口:
清單1:ResultSet 接口
我們可以使用 EasyMock 動態構建 ResultSet 接口的 Mock 對象來解決這個問題。一些簡單的測試用例只需要一個 Mock 對象,這時,我們可以用以下的方法來創建 Mock 對象:
如果需要在相對復雜的測試用例中使用多個 Mock 對象,EasyMock 提供了另外一種生成和管理 Mock 對象的機制:
IMocksControl control = EasyMock.createControl(); java.sql.Connection mockConnection = control.createMock(Connection.class); java.sql.Statement mockStatement = control.createMock(Statement.class); java.sql.ResultSet mockResultSet = control.createMock(ResultSet.class);EasyMock 類的 createControl 方法能創建一個接口 IMocksControl 的對象,該對象能創建并管理多個 Mock 對象。如果需要在測試中使用多個 Mock 對象,我們推薦您使用這一機制,因為它在多個 Mock 對象的管理上提供了相對便捷的方法。
如果您要模擬的是一個具體類而非接口,那么您需要下載擴展包 EasyMock Class Extension 2.2.2。在對具體類進行模擬時,您只要用 org.easymock.classextension.EasyMock 類中的靜態方法代替 org.easymock.EasyMock 類中的靜態方法即可。
設定 Mock 對象的預期行為和輸出
在一個完整的測試過程中,一個 Mock 對象將會經歷兩個狀態:Record 狀態和 Replay 狀態。Mock 對象一經創建,它的狀態就被置為 Record。在 Record 狀態,用戶可以設定 Mock 對象的預期行為和輸出,這些對象行為被錄制下來,保存在 Mock 對象中。
添加 Mock 對象行為的過程通常可以分為以下3步:
?? ???對 Mock 對象的特定方法作出調用;
?? ???通過 org.easymock.EasyMock 提供的靜態方法 expectLastCall 獲取上一次方法調用所對應的 IExpectationSetters 實例;
?? ???通過 IExpectationSetters 實例設定 Mock 對象的預期輸出。?
設定預期返回值
Mock 對象的行為可以簡單的理解為 Mock 對象方法的調用和方法調用所產生的輸出。在 EasyMock 2.3 中,對 Mock 對象行為的添加和設置是通過接口 IExpectationSetters 來實現的。Mock 對象方法的調用可能產生兩種類型的輸出:(1)產生返回值;(2)拋出異常。接口 IExpectationSetters 提供了多種設定預期輸出的方法,其中和設定返回值相對應的是 andReturn 方法:
mockResultSet.getString(1); expectLastCall().andReturn("My return value"); 以上的語句表示 mockResultSet 的 getString 方法被調用一次,這次調用的返回值是 "My return value"。有時,我們希望某個方法的調用總是返回一個相同的值,為了避免每次調用都為 Mock 對象的行為進行一次設定,我們可以用設置默認返回值的方法: void andStubReturn(Object value);假設我們創建了 Statement 和 ResultSet 接口的 Mock 對象 mockStatement 和 mockResultSet,在測試過程中,我們希望 mockStatement 對象的 executeQuery 方法總是返回 mockResultSet,我們可以使用如下的語句 mockStatement.executeQuery("SELECT * FROM sales_order_table"); expectLastCall().andStubReturn(mockResultSet);EasyMock 在對參數值進行匹配時,默認采用 Object.equals() 方法。因此,如果我們以 "select * from sales_order_table" 作為參數,預期方法將不會被調用。如果您希望上例中的 SQL 語句能不區分大小寫,可以用特殊的參數匹配器來解決這個問題,我們將在 "在 EasyMock 中使用參數匹配器" 一章對此進行說明。
設定預期異常拋出
對象行為的預期輸出除了可能是返回值外,還有可能是拋出異常。IExpectationSetters 提供了設定預期拋出異常的方法:
void andStubThrow(Throwable throwable);
通過以上的函數,您可以對 Mock 對象特定行為的預期輸出進行設定。除了對預期輸出進行設定,IExpectationSetters 接口還允許用戶對方法的調用次數作出限制。在 IExpectationSetters 所提供的這一類方法中,常用的一種是 times 方法: IExpectationSetters<T>times(int count);該方法可以 Mock 對象方法的調用次數進行確切的設定。假設我們希望 mockResultSet 的 getString 方法在測試過程中被調用3次,期間的返回值都是 "My return value",我們可以用如下語句:
mockResultSet.getString(1); expectLastCall().andReturn("My return value").times(3);注意到 andReturn 和 andThrow 方法的返回值依然是一個 IExpectationSetters 實例,因此我們可以在此基礎上繼續調用 times 方法。
除了設定確定的調用次數,IExpectationSetters 還提供了另外幾種設定非準確調用次數的方法:
times(int minTimes, int maxTimes):該方法最少被調用 minTimes 次,最多被調用 maxTimes 次。
atLeastOnce():該方法至少被調用一次。
anyTimes():該方法可以被調用任意次。
某些方法的返回值類型是 void,對于這一類方法,我們無需設定返回值,只要設置調用次數就可以了。以 ResultSet 接口的 close 方法為例,假設在測試過程中,該方法被調用3至5次:
expect(mockResult.close()).times(3, 5);
將 Mock 對象切換到 Replay 狀態
在生成 Mock 對象和設定 Mock 對象行為兩個階段,Mock 對象的狀態都是 Record 。在這個階段,Mock 對象會記錄用戶對預期行為和輸出的設定。
在使用 Mock 對象進行實際的測試前,我們需要將 Mock 對象的狀態切換為 Replay。在 Replay 狀態,Mock 對象能夠根據設定對特定的方法調用作出預期的響應。將 Mock 對象切換成 Replay 狀態有兩種方式,您需要根據 Mock 對象的生成方式進行選擇。如果 Mock 對象是通過 org.easymock.EasyMock 類提供的靜態方法 createMock 生成的(第1節中介紹的第一種 Mock 對象生成方法),那么 EasyMock 類提供了相應的 replay 方法用于將 Mock 對象切換為 Replay 狀態:
replay(mockResultSet);如果 Mock 對象是通過 IMocksControl 接口提供的 createMock 方法生成的(第1節中介紹的第二種Mock對象生成方法),那么您依舊可以通過 IMocksControl 接口對它所創建的所有 Mock 對象進行切換:
control.replay();
下面是示例代碼中的一個接口 SalesOrder,它的實現類 SalesOrderImpl 的主要功能是從數據庫中讀取一個 Sales Order 的 Region 和 Total Price,并根據讀取的數據計算該 Sales Order 的 Price Level(完整的實現代碼都可以在 src.zip 中找到):
清單2:SalesOrder 接口
public interface SalesOrder {……public void loadDataFromDB(ResultSet resultSet) throws SQLException; public String getPriceLevel(); }其實現類 SalesOrderImpl 中對 loadDataFromDB 的實現如下:
清單3:SalesOrderImpl 實現 public class SalesOrderImpl implements SalesOrder {......public void loadDataFromDB(ResultSet resultSet) throws SQLException{orderNumber = resultSet.getString(1);region = resultSet.getString(2);totalPrice = resultSet.getDouble(3);}...... }方法 loadDataFromDB 讀取了 ResultSet 對象包含的數據。當我們將之前定義的 Mock 對象調整為 Replay 狀態,并將該對象作為參數傳入,那么 Mock 對象的方法將會返回預先定義的預期返回值。完整的 TestCase 如下:
清單4:完整的TestCase
public class SalesOrderTestCase extends TestCase {public void testSalesOrder() {IMocksControl control = EasyMock.createControl();......ResultSet mockResultSet = control.createMock(ResultSet.class);try {......mockResultSet.next();expectLastCall().andReturn(true).times(3);expectLastCall().andReturn(false).times(1);mockResultSet.getString(1);expectLastCall().andReturn("DEMO_ORDER_001").times(1);expectLastCall().andReturn("DEMO_ORDER_002").times(1);expectLastCall().andReturn("DEMO_ORDER_003").times(1);mockResultSet.getString(2);expectLastCall().andReturn("Asia Pacific").times(1);expectLastCall().andReturn("Europe").times(1);expectLastCall().andReturn("America").times(1);mockResultSet.getDouble(3);expectLastCall().andReturn(350.0).times(1);expectLastCall().andReturn(1350.0).times(1);expectLastCall().andReturn(5350.0).times(1);control.replay();......int i = 0;String[] priceLevels = { "Level_A", "Level_C", "Level_E" };while (mockResultSet.next()) {SalesOrder order = new SalesOrderImpl();order.loadDataFromDB(mockResultSet);assertEquals(order.getPriceLevel(), priceLevels[i]);i++;}control.verify();} catch (Exception e) {e.printStackTrace();}} }在這個示例中,我們首先創建了 ResultSet 的 Mock 對象 moResultSet,并記錄該 Mock 對象的預期行為。之后我們調用了 control.replay(),將 Mock 對象的狀態置為 Replay 狀態。在實際的測試階段,Sales Order 對象的 loadDataFromDB 方法調用了 mockResultSet 對象的 getString 和 getDouble 方法讀取 mockResultSet 中的數據。Sales Order 對象根據讀取的數據計算出 Price Level,并和預期輸出進行比較。對 Mock 對象的行為進行驗證
在利用 Mock 對象進行實際的測試過程之后,我們還有一件事情沒有做:對 Mock 對象的方法調用的次數進行驗證。
為了驗證指定的方法調用真的完成了,我們需要調用 verify 方法進行驗證。和 replay 方法類似,您需要根據 Mock 對象的生成方式來選用不同的驗證方式。如果 Mock 對象是由 org.easymock.EasyMock 類提供的 createMock 靜態方法生成的,那么我們同樣采用 EasyMock 類的靜態方法 verify 進行驗證:
control.verify();
Mock 對象的重用
為了避免生成過多的 Mock 對象,EasyMock 允許對原有 Mock 對象進行重用。要對 Mock 對象重新初始化,我們可以采用 reset 方法。和 replay 和 verify 方法類似,EasyMock 提供了兩種 reset 方式:(1)如果 Mock 對象是由 org.easymock.EasyMock 類中的靜態方法 createMock 生成的,那么該 Mock 對象的可以用 EasyMock 類的靜態方法 reset 重新初始化;(2)如果 Mock 方法是由 IMocksControl 實例的 createMock 方法生成的,那么該 IMocksControl 實例方法 reset 的調用將會把所有該實例創建的 Mock 對象重新初始化。在重新初始化之后,Mock 對象的狀態將被置為 Record 狀態。
總結
以上是生活随笔為你收集整理的EasyMock 简介的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Java测试驱动开发--总结
- 下一篇: Replace Parameter wi