一、前言
相信做過開發的同學,都多多少少寫過下面的代碼,很長一段時間我一直以為這就是單元測試...
@SpringBootTest
@RunWith(SpringRunner.class)
public class UnitTest1 {
@Autowired
private UnitService unitService;
@Test
public void test() {
System.out.println("----------------------");
System.out.println(unitService.sayHello());
System.out.println("----------------------");
}
}
但這是單元測試嘛?unitService 中可能還依賴了 Dao 的操作;如果是微服務,可能還要起注冊中心。那么這個“單元”也太大了吧!如果把它稱為集成測試,可能更恰當一點,那么有沒有可能最小粒度進行單元測試嘛?
單元測試應該是一個帶有隔離性的功能測試。在單元測試中,應盡量避免其他類或系統的副作用影響。
單元測試的目標是一小段代碼,例如方法或類。方法或類的外部依賴關系應從單元測試中移除,而改為測試框架創建的 mock 對象來替換依賴對象。
單元測試一般由開發人員編寫,通過驗證或斷言目標的一些行為或狀態來達到測試的目的。
二、JUnit 框架
JUnit 是一個測試框架,它使用注解來標識測試方法。JUnit 是 Github 上托管的一個開源項目。
一個 JUnit 測試指的是一個包含在測試類中的方法,要定義某個方法為測試方法,請使用 @Test 注解標注該方法。該方法執行被測代碼,可以使用 JUnit 或另一個 Assert 框架提供的 assert 方法來檢查預期結果與實際結果是否一致,這些方法調用通常稱為斷言或斷言語句。
public class UnitTest2 {
@Test
public void test() {
String sayHello = "Hello World";
Assert.assertEquals("Hello World", sayHello);
}
}
以下是一些常用的 JUnit 注解:
注解 | 描述 |
---|---|
@Test | 將方法標識為測試方法 |
@Before | 在每次測試之前執行。用於准備測試環境(例如,讀取輸入數據,初始化類) |
@After | 每次測試之后執行。用於清理測試環境(例如,刪除臨時數據,恢復默認值) |
@BeforeClass | 用於 static方法,在所有測試開始之前執行一次。它用於執行耗時的活動,例如:連接到數據庫 |
@AfterClass | 用於 static方法,在完成所有測試之后,執行一次。它用於執行清理活動,例如:與數據庫斷開連接 |
@Ignore | 指定要忽略的測試 |
@Test(expected = Exception.class) | 如果該方法未引發命名異常,則失敗 |
@Test(timeout=100) | 如果該方法花費的時間超過100毫秒,則失敗 |
以下是一些常用的 Assert 斷言:
聲明 | 描述 |
---|---|
fail([message]) | 使方法失敗。在執行測試代碼之前,可用於檢查未到達代碼的特定部分或測試失敗 |
assertTrue([message,]布爾條件) | 檢查布爾條件是否為真 |
assertFalse([message,]布爾條件) | 檢查布爾條件是否為假 |
assertEquals([message,]預期,實際) | 測試兩個值是否相同。注意:對於數組,會檢查引用而不是數組的內容 |
assertNull([message,]對象) | 檢查對象是否為空 |
assertNotNull([message,]對象) | 檢查對象是否不為空 |
assertSame([message,]預期,實際) | 檢查兩個變量是否引用同一對象 |
assertNotSame([message,]預期,實際) | 檢查兩個變量是否引用了不同的對象 |
三、Mockito 框架
從上面的介紹我們可以認識到,如何減少對外部的依賴才是實踐單元測試的關鍵。而這正是 Mockito 的使命,Mockito 是一個流行的 mock 框架,可以與 JUnit 結合使用,Mockito 允許我們創建和配置 mock 對象,使用 Mockito 將大大簡化了具有外部依賴項的類的測試開發。spring-boot-starter-test 中默認集成了 Mockito,不需要額外引入。
在測試中使用 Mockito,通常會:
- mock 外部依賴關系並將 mock 對象插入待測代碼
- 執行被測代碼
- 驗證代碼是否正確執行
3.1 使用 Mockito 創建 mock 對象
Mockito 提供了幾種創建 mock 對象的方法:
- 使用靜態 mock() 方法
- 使用 @Mock 注解
如果使用 @Mock 注解,則必須觸發創建帶有 @Mock 注解的對象。使用 MockitoRule 可以做到,它通過調用靜態方法 MockitoAnnotations.initMocks(this) 來填充帶 @Mock 注解的字段。或者可以使用 @RunWith(MockitoJUnitRunner.class)。
public class UnitTest3 {
// 觸發創建帶有 @Mock 注解的對象
@Rule public MockitoRule mockitoRule = MockitoJUnit.rule();
// 1. 使用 @Mock 注解創建 mock 對象
@Mock private UnitDao unitDao;
@Test
public void test() {
// 2. 使用靜態 mock() 方法創建 mock 對象
Iterator iterator = mock(Iterator.class);
// when...thenReturn / doReturn...when 模擬依賴調用
when(iterator.next()).thenReturn("hello");
doReturn(1).when(unitDao).delete(anyLong());
// 斷言
Assert.assertEquals("hello", iterator.next());
Assert.assertEquals(new Integer(1), unitDao.delete(1L));
}
}
3.2 使用 mock 對象實踐單元測試
我們要單元測試的內容,常常包含着對數據庫的訪問等等,那么我們要如何 mock 掉這部分調用呢?我們可以使用 @InjectMocks 注解創建實例並使用 mock 對象進行依賴注入。
@Service
public class UnitServiceImpl implements UnitService {
@Autowired
private UnitDao unitDao;
@Override
public String sayHello() {
Integer delete = unitDao.delete(1L);
System.out.println(delete);
return "hello unit";
}
}
@RunWith(MockitoJUnitRunner.class)
public class UnitTest2 {
@Mock
private UnitDao unitDao;
@InjectMocks
private UnitServiceImpl unitService;
@Test
public void unitTest() {
// mock 調用
when(unitDao.delete(anyLong())).thenReturn(1);
Assert.assertEquals("hello unit", unitService.sayHello());
}
}
Mockito 還有很多有趣的實踐,比如:@Spy或spy()方法、verify()驗證等等,鑒於篇幅原因,讀者可自行挖掘。
3.3 使用 PowerMock mock 靜態方法。
Mockito 也有一些局限性。例如:不能 mock 靜態方法和私有方法。有關詳細信息,請參閱 Mockito限制的常見問題解答。這個時候我們就要用到 PowerMock,PowerMock 支持 JUnit 和 TestNG,擴展了 EasyMock 和 Mockito 框架,增加了mock static、final 方法的功能。
首先需要引入 PowerMock 的依賴:
<!-- PowerMock -->
<dependency>
<groupId>org.powermock</groupId>
<artifactId>powermock-module-junit4</artifactId>
<version>2.0.7</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.powermock</groupId>
<artifactId>powermock-api-mockito2</artifactId>
<version>2.0.7</version>
</dependency>
接下來就能愉快的 mock 靜態方法了。
@RunWith(PowerMockRunner.class)
@PrepareForTest({StringUtils.class})
public class UnitTest4 {
@Test
public void test() {
mockStatic(StringUtils.class);
when(StringUtils.getFilename(anyString())).thenReturn("localhost");
Assert.assertEquals("localhost", StringUtils.getFilename(""));
}
}