日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 >

Java编程技巧之单元测试用例编写流程

發布時間:2024/8/23 41 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Java编程技巧之单元测试用例编写流程 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

簡介:?立足于“如何來編寫單元測試用例”,讓大家“有章可循”,快速編寫出單元測試用例。

作者 | 常意
來源 | 阿里技術公眾號

溫馨提示:本文較長,同學們可收藏后再看 :)

前言

清代杰出思想家章學誠有一句名言:“學必求其心得,業必貴其專精。

意思是:學習上一定要追求心得體會,事業上一定要貴以專注精深。做技術就是這樣,一件事如果做到了極致,就必然會有所心得體會。作者最近在一個項目上,追求單元測試覆蓋率到極致,所以才有了這篇心得體會。

上一篇文章《Java單元測試技巧之PowerMock》除了介紹單元測試基礎知識外,主要介紹了“為什么要編寫單元測試”。很多同學讀完后,還是不能快速地編寫單元測試用例。而這篇文章,立足于“如何來編寫單元測試用例”,能夠讓同學們“有章可循”,能快速地編寫出單元測試用例。

1.編寫單元測試用例

1.1.測試框架簡介

Mockito是一個單元測試模擬框架,可以讓你寫出優雅、簡潔的單元測試代碼。Mockito采用了模擬技術,模擬了一些在應用中依賴的復雜對象,從而把測試對象和依賴對象隔離開來。

PowerMock是一個單元測試模擬框架,是在其它單元測試模擬框架的基礎上做出擴展。 通過提供定制的類加載器以及一些字節碼篡改技術的應用,PowerMock實現了對靜態方法、構造方法、私有方法以及final方法的模擬支持等強大的功能。但是,正因為PowerMock進行了字節碼篡改,導致部分單元測試用例并不被JaCoco統計覆蓋率。

通過作者多年單元測試的編寫經驗,優先推薦使用Mockito提供的功能;只有在Mockito提供的功能不能滿足需求時,才會采用PowerMock提供的功能;但是,不推薦使用影響JaCoco統計覆蓋率的PowerMock功能。在本文中,我們也不會對影響JaCoco統計覆蓋率的PowerMock功能進行介紹。

下面,將以Mockito為主、以PowerMock為輔,介紹一下如何編寫單元測試用例。

1.2.測試框架引入

為了引入Mockito和PowerMock包,需要在maven項目的pom.xml文件中加入以下包依賴:

其中,powermock.version為2.0.9,為當前的最新版本,可根據實際情況修改。在PowerMock包中,已經包含了對應的Mockito和JUnit包,所以無需單獨引入Mockito和JUnit包。

1.3.典型代碼案例

一個典型的服務代碼案例如下:

/*** 用戶服務類*/ @Service public class UserService {/** 服務相關 *//** 用戶DAO */@Autowiredprivate UserDAO userDAO;/** 標識生成器 */@Autowiredprivate IdGenerator idGenerator;/** 參數相關 *//** 可以修改 */@Value("${userService.canModify}")private Boolean canModify;/*** 創建用戶* * @param userCreate 用戶創建* @return 用戶標識*/public Long createUser(UserVO userCreate) {// 獲取用戶標識Long userId = userDAO.getIdByName(userCreate.getName());// 根據存在處理// 根據存在處理: 不存在則創建if (Objects.isNull(userId)) {userId = idGenerator.next();UserDO create = new UserDO();create.setId(userId);create.setName(userCreate.getName());userDAO.create(create);}// 根據存在處理: 已存在可修改else if (Boolean.TRUE.equals(canModify)) {UserDO modify = new UserDO();modify.setId(userId);modify.setName(userCreate.getName());userDAO.modify(modify);}// 根據存在處理: 已存在禁修改else {throw new UnsupportedOperationException("不支持修改");}// 返回用戶標識return userId;} }

1.4.測試用例編寫

采用Mockito和PowerMock單元測試模擬框架,編寫的單元測試用例如下:

UserServiceTest.java:

/*** 用戶服務測試類*/ @RunWith(PowerMockRunner.class) public class UserServiceTest {/** 模擬依賴對象 *//** 用戶DAO */@Mockprivate UserDAO userDAO;/** 標識生成器 */@Mockprivate IdGenerator idGenerator;/** 定義被測對象 *//** 用戶服務 */@InjectMocksprivate UserService userService;/*** 在測試之前*/@Beforepublic void beforeTest() {// 注入依賴對象Whitebox.setInternalState(userService, "canModify", Boolean.TRUE);}/*** 測試: 創建用戶-新*/@Testpublic void testCreateUserWithNew() {// 模擬依賴方法// 模擬依賴方法: userDAO.getByNameMockito.doReturn(null).when(userDAO).getIdByName(Mockito.anyString());// 模擬依賴方法: idGenerator.nextLong userId = 1L;Mockito.doReturn(userId).when(idGenerator).next();// 調用被測方法String text = ResourceHelper.getResourceAsString(getClass(), "userCreateVO.json");UserVO userCreate = JSON.parseObject(text, UserVO.class);Assert.assertEquals("用戶標識不一致", userId, userService.createUser(userCreate));// 驗證依賴方法// 驗證依賴方法: userDAO.getByNameMockito.verify(userDAO).getIdByName(userCreate.getName());// 驗證依賴方法: idGenerator.nextMockito.verify(idGenerator).next();// 驗證依賴方法: userDAO.createArgumentCaptor < UserDO> userCreateCaptor = ArgumentCaptor.forClass(UserDO.class);Mockito.verify(userDAO).create(userCreateCaptor.capture());text = ResourceHelper.getResourceAsString(getClass(), "userCreateDO.json");Assert.assertEquals("用戶創建不一致", text, JSON.toJSONString(userCreateCaptor.getValue()));// 驗證依賴對象Mockito.verifyNoMoreInteractions(idGenerator, userDAO);}/*** 測試: 創建用戶-舊*/@Testpublic void testCreateUserWithOld() {// 模擬依賴方法// 模擬依賴方法: userDAO.getByNameLong userId = 1L;Mockito.doReturn(userId).when(userDAO).getIdByName(Mockito.anyString());// 調用被測方法String text = ResourceHelper.getResourceAsString(getClass(), "userCreateVO.json");UserVO userCreate = JSON.parseObject(text, UserVO.class);Assert.assertEquals("用戶標識不一致", userId, userService.createUser(userCreate));// 驗證依賴方法// 驗證依賴方法: userDAO.getByNameMockito.verify(userDAO).getIdByName(userCreate.getName());// 驗證依賴方法: userDAO.modifyArgumentCaptor < UserDO> userModifyCaptor = ArgumentCaptor.forClass(UserDO.class);Mockito.verify(userDAO).modify(userModifyCaptor.capture());text = ResourceHelper.getResourceAsString(getClass(), "userModifyDO.json");Assert.assertEquals("用戶修改不一致", text, JSON.toJSONString(userModifyCaptor.getValue()));// 驗證依賴對象Mockito.verifyNoInteractions(idGenerator);Mockito.verifyNoMoreInteractions(userDAO);}/*** 測試: 創建用戶-異常*/@Testpublic void testCreateUserWithException() {// 注入依賴對象Whitebox.setInternalState(userService, "canModify", Boolean.FALSE);// 模擬依賴方法// 模擬依賴方法: userDAO.getByNameLong userId = 1L;Mockito.doReturn(userId).when(userDAO).getIdByName(Mockito.anyString());// 調用被測方法String text = ResourceHelper.getResourceAsString(getClass(), "userCreateVO.json");UserVO userCreate = JSON.parseObject(text, UserVO.class);UnsupportedOperationException exception = Assert.assertThrows("返回異常不一致",UnsupportedOperationException.class, () -> userService.createUser(userCreate));Assert.assertEquals("異常消息不一致", "不支持修改", exception.getMessage());} }

userCreateVO.json:

{"name":"test"}

userCreateDO.json:

{"id":1,"name":"test"}

userModifyDO.json:

{"id":1,"name":"test"}

通過執行以上測試用例,可以看到對源代碼進行了100%的行覆蓋。

2.測試用例編寫流程

通過上一章編寫Java類單元測試用例的實踐,可以總結出以下Java類單元測試用例的編寫流程:

單元測試用例編寫流程

?

上面一共有3個測試用例,這里僅以測試用例testCreateUserWithNew(測試: 創建用戶-新)為例說明。

2.1.定義對象階段

第1步是定義對象階段,主要包括定義被測對象、模擬依賴對象(類成員)、注入依賴對象(類成員)3大部分。

2.1.1.定義被測對象

在編寫單元測試時,首先需要定義被測對象,或直接初始化、或通過Spy包裝……其實,就是把被測試服務類進行實例化。

/** 定義被測對象 */ /** 用戶服務 */ @InjectMocks private UserService userService;

2.1.2.模擬依賴對象(類成員)

在一個服務類中,我們定義了一些類成員對象——服務(Service)、數據訪問對象(DAO)、參數(Value)等。在Spring框架中,這些類成員對象通過@Autowired、@Value等方式注入,它們可能涉及復雜的環境配置、依賴第三方接口服務……但是,在單元測試中,為了解除對這些類成員對象的依賴,我們需要對這些類成員對象進行模擬。

/** 模擬依賴對象 */ /** 用戶DAO */ @Mock private UserDAO userDAO; /** 標識生成器 */ @Mock private IdGenerator idGenerator;

2.1.3.注入依賴對象(類成員)

當模擬完這些類成員對象后,我們需要把這些類成員對象注入到被測試類的實例中。以便在調用被測試方法時,可能使用這些類成員對象,而不至于拋出空指針異常。

/** 定義被測對象 */ /** 用戶服務 */ @InjectMocks private UserService userService;/*** 在測試之前*/ @Before public void beforeTest() {// 注入依賴對象Whitebox.setInternalState(userService, "canModify", Boolean.TRUE); }

2.2.模擬方法階段

第2步是模擬方法階段,主要包括模擬依賴對象(參數或返回值)、模擬依賴方法2大部分。

2.2.1.模擬依賴對象(參數或返回值)

通常,在調用一個方法時,需要先指定方法的參數,然后獲取到方法的返回值。所以,在模擬方法之前,需要先模擬該方法的參數和返回值。

Long userId = 1L;

2.2.2.模擬依賴方法

在模擬完依賴的參數和返回值后,就可以利用Mockito和PowerMock的功能,進行依賴方法的模擬。如果依賴對象還有方法調用,還需要模擬這些依賴對象的方法。

// 模擬依賴方法 // 模擬依賴方法: userDAO.getByName Mockito.doReturn(null).when(userDAO).getIdByName(Mockito.anyString()); // 模擬依賴方法: idGenerator.next Mockito.doReturn(userId).when(idGenerator).next();

2.3.調用方法階段

第3步是調用方法階段,主要包括模擬依賴對象(參數)、調用被測方法、驗證參數對象(返回值)3步。

2.3.1.模擬依賴對象(參數)

在調用被測方法之前,需要模擬被測方法的參數。如果這些參數還有方法調用,還需要模擬這些參數的方法。

String text = ResourceHelper.getResourceAsString(getClass(), "userCreateVO.json"); UserVO userCreate = JSON.parseObject(text, UserVO.class);

2.3.2.調用被測方法

在準備好參數對象后,就可以調用被測試方法了。如果被測試方法有返回值,需要定義變量接收返回值;如果被測試方法要拋出異常,需要指定期望的異常。

userService.createUser(userCreate)

2.3.3.驗證數據對象(返回值)

在調用被測試方法后,如果被測試方法有返回值,需要驗證這個返回值是否符合預期;如果被測試方法要拋出異常,需要驗證這個異常是否滿足要求。

Assert.assertEquals("用戶標識不一致", userId, userService.createUser(userCreate));

2.4.驗證方法階段

第4步是驗證方法階段,主要包括驗證依賴方法、驗證數據對象(參數)、驗證依賴對象3步。

2.4.1.驗證依賴方法

作為一個完整的測試用例,需要對每一個模擬的依賴方法調用進行驗證。

// 驗證依賴方法 // 驗證依賴方法: userDAO.getByName Mockito.verify(userDAO).getIdByName(userCreate.getName()); // 驗證依賴方法: idGenerator.next Mockito.verify(idGenerator).next(); // 驗證依賴方法: userDAO.create ArgumentCaptor < UserDO> userCreateCaptor = ArgumentCaptor.forClass(UserDO.class); Mockito.verify(userDAO).create(userCreateCaptor.capture());

2.4.2.驗證數據對象(參數)

對應一些模擬的依賴方法,有些參數對象是被測試方法內部生成的。為了驗證代碼邏輯的正確性,就需要對這些參數對象進行驗證,看這些參數對象值是否符合預期。

text = ResourceHelper.getResourceAsString(getClass(), "userCreateDO.json"); Assert.assertEquals("用戶創建不一致", text, JSON.toJSONString(userCreateCaptor.getValue()));

2.4.3.驗證依賴對象

作為一個完整的測試用例,應該保證每一個模擬的依賴方法調用都進行了驗證。正好,Mockito提供了一套方法,用于驗證模擬對象所有方法調用都得到了驗證。

// 驗證依賴對象 Mockito.verifyNoMoreInteractions(idGenerator, userDAO);

3.定義被測對象

在編寫單元測試時,首先需要定義被測對象,或直接初始化、或通過Spy包裝……其實,就是把被測試服務類進行實例化。

3.1.直接構建對象

直接構建一個對象,總是簡單又直接。

UserService userService = new UserService();

3.2.利用Mockito.spy方法

Mockito提供一個spy功能,用于攔截那些尚未實現或不期望被真實調用的方法,默認所有方法都是真實方法,除非主動去模擬對應方法。所以,利用spy功能來定義被測對象,適合于需要模擬被測類自身方法的情況,適用于普通類、接口和虛基類。

UserService userService = Mockito.spy(new UserService()); UserService userService = Mockito.spy(UserService.class); AbstractOssService ossService = Mockito.spy(AbstractOssService.class);

3.3.利用@Spy注解

@Spy注解跟Mockito.spy方法一樣,可以用來定義被測對象,適合于需要模擬被測類自身方法的情況,適用于普通類、接口和虛基類。@Spy注解需要配合@RunWith注解使用。

@RunWith(PowerMockRunner.class) public class CompanyServiceTest {@Spyprivate UserService userService = new UserService();... }

注意:@Spy注解對象需要初始化。如果是虛基類或接口,可以用Mockito.mock方法實例化。

3.4.利用@InjectMocks注解

@InjectMocks注解用來創建一個實例,并將其它對象(@Mock、@Spy或直接定義的對象)注入到該實例中。所以,@InjectMocks注解本身就可以用來定義被測對象。@InjectMocks注解需要配合@RunWith注解使用。

@RunWith(PowerMockRunner.class) public class UserServiceTest {@InjectMocksprivate UserService userService;... }

4.模擬依賴對象

在編寫單元測試用例時,需要模擬各種依賴對象——類成員、方法參數和方法返回值。

4.1.直接構建對象

如果需要構建一個對象,最簡單直接的方法就是——定義對象并賦值。

Long userId = 1L; String userName = "admin"; UserDO user = new User(); user.setId(userId); user.setName(userName); List < Long> userIdList = Arrays.asList(1L, 2L, 3L);

4.2.反序列化對象

如果對象字段或層級非常龐大,采用直接構建對象方法,可能會編寫大量構建程序代碼。這種情況,可以考慮反序列化對象,將會大大減少程序代碼。由于JSON字符串可讀性高,這里就以JSON為例,介紹反序列化對象。

反序列化模型對象:

String text = ResourceHelper.getResourceAsString(getClass(), "user.json"); UserDO user = JSON.parseObject(text, UserDO.class);

反序列化集合對象:

String text = ResourceHelper.getResourceAsString(getClass(), "userList.json"); List < UserDO> userList = JSON.parseArray(text, UserDO.class);

反序列化映射對象:

String text = ResourceHelper.getResourceAsString(getClass(), "userMap.json"); Map < Long, UserDO> userMap = JSON.parseObject(text, new TypeReference < Map < Long, UserDO>>() {});

4.3.利用Mockito.mock方法

Mockito提供一個mock功能,用于攔截那些尚未實現或不期望被真實調用的方法,默認所有方法都已被模擬——方法為空并返回默認值(null或0),除非主動執行doCallRealMethod或thenCallRealMethod操作,才能夠調用真實的方法。

利用Mockito.mock方法模擬依賴對象,主要用于以下幾種情形:

  • 只使用類實例,不使用類屬性;
  • 類屬性太多,但使用其中少量屬性(可以mock屬性返回值);
  • 類是接口或虛基類,并不關心其具體實現類。
  • MockClass mockClass = Mockito.mock(MockClass.class); List < Long> userIdList = (List < Long>)Mockito.mock(List.class);

    4.4.利用@Mock注解

    @Mock注解跟Mockito.mock方法一樣,可以用來模擬依賴對象,適用于普通類、接口和虛基類。@Mock注解需要配合@RunWith注解使用。

    @RunWith(PowerMockRunner.class) public class UserServiceTest {@Mockprivate UserDAO userDAO;... }

    4.5.利用Mockito.spy方法

    Mockito.spy方法跟Mockito.mock方法功能相似,只是Mockito.spy方法默認所有方法都是真實方法,除非主動去模擬對應方法。

    UserService userService = Mockito.spy(new UserService()); UserService userService = Mockito.spy(UserService.class); AbstractOssService ossService = Mockito.spy(AbstractOssService.class);

    4.6.利用@Spy注解

    @Spy注解跟Mockito.spy方法一樣,可以用來模擬依賴對象,適用于普通類、接口和虛基類。@Spy注解需要配合@RunWith注解使用。

    @RunWith(PowerMockRunner.class) public class CompanyServiceTest {@Spyprivate UserService userService = new UserService();... }

    注意:@Spy注解對象需要初始化。如果是虛基類或接口,可以用Mockito.mock方法實例化。

    5.注入依賴對象

    當模擬完這些類成員對象后,我們需要把這些類成員對象注入到被測試類的實例中。以便在調用被測試方法時,可能使用這些類成員對象,而不至于拋出空指針異常。

    5.1.利用Setter方法注入

    如果類定義了Setter方法,可以直接調用方法設置字段值。

    userService.setMaxCount(100); userService.setUserDAO(userDAO);

    5.2.利用ReflectionTestUtils.setField方法注入

    JUnit提供ReflectionTestUtils.setField方法設置屬性字段值。

    ReflectionTestUtils.setField(userService, "maxCount", 100); ReflectionTestUtils.setField(userService, "userDAO", userDAO);

    5.3.利用Whitebox.setInternalState方法注入

    PowerMock提供Whitebox.setInternalState方法設置屬性字段值。

    Whitebox.setInternalState(userService, "maxCount", 100); Whitebox.setInternalState(userService, "userDAO", userDAO);

    5.4.利用@InjectMocks注解注入

    @InjectMocks注解用來創建一個實例,并將其它對象(@Mock、@Spy或直接定義的對象)注入到該實例中。@InjectMocks注解需要配合@RunWith注解使用。

    @RunWith(PowerMockRunner.class) public class UserServiceTest {@Mockprivate UserDAO userDAO;private Boolean canModify;@InjectMocksprivate UserService userService;... }

    5.5.設置靜態常量字段值

    有時候,我們需要對靜態常量對象進行模擬,然后去驗證是否執行了對應分支下的方法。比如:需要模擬Lombok的@Slf4j生成的log靜態常量。但是,Whitebox.setInternalState方法和@InjectMocks注解并不支持設置靜態常量,需要自己實現一個設置靜態常量的方法:

    public final class FieldHelper {public static void setStaticFinalField(Class< ?> clazz, String fieldName, Object fieldValue) throws NoSuchFieldException, IllegalAccessException {Field field = clazz.getDeclaredField(fieldName);FieldUtils.removeFinalModifier(field);FieldUtils.writeStaticField(field, fieldValue, true);} }

    具體使用方法如下:

    FieldHelper.setStaticFinalField(UserService.class, "log", log);

    注意:經過測試,該方法對于int、Integer等基礎類型并不生效,應該是編譯器常量優化導致。

    6.模擬依賴方法

    在模擬完依賴的參數和返回值后,就可以利用Mockito和PowerMock的功能,進行依賴方法的模擬。如果依賴對象還有方法調用,還需要模擬這些依賴對象的方法。

    6.1.根據返回模擬方法

    6.1.1.模擬無返回值方法

    Mockito.doNothing().when(userDAO).delete(userId);

    6.1.2.模擬方法單個返回值

    Mockito.doReturn(user).when(userDAO).get(userId); Mockito.when(userDAO.get(userId)).thenReturn(user);

    6.1.3.模擬方法多個返回值

    直接列舉出多個返回值:

    Mockito.doReturn(record0, record1, record2, null).when(recordReader).read(); Mockito.when(recordReader.read()).thenReturn(record0, record1, record2, null);

    轉化列表為多個返回值:

    List< Record> recordList = ...; Mockito.doReturn(recordList.get(0), recordList.subList(1, recordList.size()).toArray()).when(recordReader).read(); Mockito.when(recordReader.read()).thenReturn(recordList.get(0), recordList.subList(1, recordList.size()).toArray());

    6.1.4.模擬方法定制返回值

    可利用Answer定制方法返回值:

    Map< Long, UserDO> userMap = ...; Mockito.doAnswer(invocation -> userMap.get(invocation.getArgument(0))).when(userDAO).get(Mockito.anyLong()); Mockito.when(userDAO.get(Mockito.anyLong())).thenReturn(invocation -> userMap.get(invocation.getArgument(0))); Mockito.when(userDAO.get(Mockito.anyLong())).then(invocation -> userMap.get(invocation.getArgument(0)));

    6.1.5.模擬方法拋出單個異常

    指定單個異常類型:

    Mockito.doThrow(PersistenceException.class).when(userDAO).get(Mockito.anyLong()); Mockito.when(userDAO.get(Mockito.anyLong())).thenThrow(PersistenceException.class);

    指定單個異常對象:

    Mockito.doThrow(exception).when(userDAO).get(Mockito.anyLong()); Mockito.when(userDAO.get(Mockito.anyLong())).thenThrow(exception);

    6.1.6.模擬方法拋出多個異常

    指定多個異常類型:

    Mockito.doThrow(PersistenceException.class, RuntimeException.class).when(userDAO).get(Mockito.anyLong()); Mockito.when(userDAO.get(Mockito.anyLong())).thenThrow(PersistenceException.class, RuntimeException.class);

    指定多個異常對象:

    Mockito.doThrow(exception1, exception2).when(userDAO).get(Mockito.anyLong()); Mockito.when(userDAO.get(Mockito.anyLong())).thenThrow(exception1, exception2);

    6.1.7.直接調用真實方法

    Mockito.doCallRealMethod().when(userService).getUser(userId); Mockito.when(userService.getUser(userId)).thenCallRealMethod();

    6.2.根據參數模擬方法

    Mockito提供do-when語句和when-then語句模擬方法。

    6.2.1.模擬無參數方法

    對于無參數的方法模擬:

    Mockito.doReturn(deleteCount).when(userDAO).deleteAll(); Mockito.when(userDAO.deleteAll()).thenReturn(deleteCount);

    6.2.2.模擬指定參數方法

    對于指定參數的方法模擬:

    Mockito.doReturn(user).when(userDAO).get(userId); Mockito.when(userDAO.get(userId)).thenReturn(user);

    6.2.3.模擬任意參數方法

    在編寫單元測試用例時,有時候并不關心傳入參數的具體值,可以使用Mockito參數匹配器的any方法。Mockito提供了anyInt、anyLong、anyString、anyList、anySet、anyMap、any(Class clazz)等方法來表示任意值。

    Mockito.doReturn(user).when(userDAO).get(Mockito.anyLong()); Mockito.when(userDAO.get(Mockito.anyLong())).thenReturn(user);

    6.2.4.模擬可空參數方法

    Mockito參數匹配器的any具體方法,并不能夠匹配null對象。而Mockito提供一個nullable方法,可以匹配包含null對象的任意對象。此外,Mockito.any()方法也可以用來匹配可空參數。

    Mockito.doReturn(user).when(userDAO).queryCompany(Mockito.anyLong(), Mockito.nullable(Long.class)); Mockito.when(userDAO.queryCompany(Mockito.anyLong(), Mockito< Long>.any())).thenReturn(user);

    6.2.5.模擬必空參數方法

    同樣,如果要匹配null對象,可以使用isNull方法,或使用eq(null)。

    Mockito.doReturn(user).when(userDAO).queryCompany(Mockito.anyLong(), Mockito.isNull()); Mockito.when(userDAO.queryCompany(Mockito.anyLong(), Mockito.eq(null))).thenReturn(user);

    6.2.6.模擬不同參數方法

    Mockito支持按不同的參數分別模擬同一方法。

    Mockito.doReturn(user1).when(userDAO).get(1L); Mockito.doReturn(user2).when(userDAO).get(2L); ...

    注意:如果一個參數滿足多個模擬方法條件,會以最后一個模擬方法為準。

    6.2.7.模擬可變參數方法

    對于一些變長度參數方法,可以按實際參數個數進行模擬:

    Mockito.when(userService.delete(Mockito.anyLong()).thenReturn(true); Mockito.when(userService.delete(1L, 2L, 3L).thenReturn(true);

    也可以用Mockito.any()模擬一個通用匹配方法:

    Mockito.when(userService.delete(Mockito.< Long>any()).thenReturn(true);

    注意:Mockito.< T>any()并不等于Mockito.any(Class< T> type),前者可以匹配null和類型T的可變參數,后者只能匹配T必填參數。

    6.3.模擬其它特殊方法

    6.3.1.模擬final方法

    PowerMock提供對final方法的模擬,方法跟模擬普通方法一樣。但是,需要把對應的模擬類添加到@PrepareForTest注解中。

    // 添加@PrepareForTest注解 @PrepareForTest({UserService.class})// 跟模擬普通方法完全一致 Mockito.doReturn(userId).when(idGenerator).next(); Mockito.when(idGenerator.next()).thenReturn(userId);

    6.3.2.模擬私有方法

    PowerMock提供提對私有方法的模擬,但是需要把私有方法所在的類放在@PrepareForTest注解中。

    PowerMockito.doReturn(true).when(UserService.class, "isSuper", userId); PowerMockito.when(UserService.class, "isSuper", userId).thenReturn(true);

    6.3.3.模擬構造方法

    PowerMock提供PowerMockito.whenNew方法來模擬構造方法,但是需要把使用構造方法的類放在@PrepareForTest注解中。

    PowerMockito.whenNew(UserDO.class).withNoArguments().thenReturn(userDO); PowerMockito.whenNew(UserDO.class).withArguments(userId, userName).thenReturn(userDO);

    6.3.4.模擬靜態方法

    PowerMock提供PowerMockito.mockStatic和PowerMockito.spy來模擬靜態方法類,然后就可以模擬靜態方法了。同樣,需要把對應的模擬類添加到@PrepareForTest注解中。

    // 模擬對應的類 PowerMockito.mockStatic(HttpHelper.class); PowerMockito.spy(HttpHelper.class);// 模擬對應的方法 PowerMockito.when(HttpHelper.httpPost(SERVER_URL)).thenReturn(response); PowerMockito.doReturn(response).when(HttpHelper.class, "httpPost", SERVER_URL); PowerMockito.when(HttpHelper.class, "httpPost", SERVER_URL).thenReturn(response);

    注意:第一種方式不適用于PowerMockito.spy模擬的靜態方法類。

    7.調用被測方法

    在準備好參數對象后,就可以調用被測試方法了。

    如果把方法按訪問權限分類,可以簡單地分為有訪問權限無訪問權限兩種。但實際上,Java語言中提供了public、protected、private和缺失共4種權限修飾符,在不同的環境下又對應不同的訪問權限。具體映射關系如下:

    修飾符本類本包子類其它
    public
    protected
    缺省
    private

    下面,將根據有訪問權限無訪問權限兩種情況,來介紹如何調用被測方法。

    7.1.調用構造方法

    7.1.1.調用有訪問權限的構造方法

    可以直接調用有訪問權限的構造方法。

    UserDO user = new User(); UserDO user = new User(1L, "admin");

    7.1.2.調用無訪問權限的構造方法

    調用無訪問權限的構造方法,可以使用PowerMock提供的Whitebox.invokeConstructor方法。

    Whitebox.invokeConstructor(NumberHelper.class); Whitebox.invokeConstructor(User.class, 1L, "admin");

    備注:該方法也可以調用有訪問權限的構造方法,但是不建議使用。

    7.2.調用普通方法

    7.2.1.調用有訪問權限的普通方法

    可以直接調用有訪問權限的普通方法。

    userService.deleteUser(userId); User user = userService.getUser(userId);

    7.2.2.調用無權限訪問的普通方法

    調用無訪問權限的普通方法,可以使用PowerMock提供的Whitebox.invokeMethod方法。

    User user = (User)Whitebox.invokeMethod(userService, "isSuper", userId);

    也可以使用PowerMock提供Whitebox.getMethod方法和PowerMockito.method方法,可以直接獲取對應類方法對象。然后,通過Method的invoke方法,可以調用沒有訪問權限的方法。

    Method method = Whitebox.getMethod(UserService.class, "isSuper", Long.class); Method method = PowerMockito.method(UserService.class, "isSuper", Long.class); User user = (User)method.invoke(userService, userId);

    備注:該方法也可以調用有訪問權限的普通方法,但是不建議使用。

    7.3.調用靜態方法

    7.3.1.調用有權限訪問的靜態方法

    可以直接調用有訪問權限的靜態方法。

    boolean isPositive = NumberHelper.isPositive(-1);

    7.3.2.調用無權限訪問的靜態方法

    調用無權限訪問的靜態方法,可以使用PowerMock提供的Whitebox.invokeMethod方法。

    String value = (String)Whitebox.invokeMethod(JSON.class, "toJSONString", object);

    備注:該方法也可以調用有訪問權限的靜態方法,但是不建議使用。

    8.驗證依賴方法

    在單元測試中,驗證是確認模擬的依賴方法是否按照預期被調用或未調用的過程。Mockito提供了許多方法來驗證依賴方法調用,給我們編寫單元測試用例帶來了很大的幫助。

    8.1.根據參數驗證方法調用

    8.1.1.驗證無參數方法調用

    Mockito.verify(userDAO).deleteAll();

    8.1.2.驗證指定參數方法調用

    Mockito.verify(userDAO).delete(userId); Mockito.verify(userDAO).delete(Mockito.eq(userId));

    8.1.3.驗證任意參數方法調用

    Mockito.verify(userDAO).delete(Mockito.anyLong());

    8.1.4.驗證可空參數方法調用

    Mockito.verify(userDAO).queryCompany(Mockito.anyLong(), Mockito.nullable(Long.class));

    8.1.5.驗證必空參數方法調用

    Mockito.verify(userDAO).queryCompany(Mockito.anyLong(), Mockito.isNull());

    8.1.6.驗證可變參數方法調用

    對于一些變長度參數方法,可以按實際參數個數進行驗證:

    Mockito.verify(userService).delete(Mockito.any(Long.class)); Mockito.verify(userService).delete(1L, 2L, 3L);

    也可以用Mockito.any()進行通用驗證:

    Mockito.verify(userService).delete(Mockito.< Long>any());

    8.2.驗證方法調用次數

    8.2.1.驗證方法默認調用1次

    Mockito.verify(userDAO).delete(userId);

    8.2.2.驗證方法從不調用

    Mockito.verify(userDAO, Mockito.never()).delete(userId);

    8.2.3.驗證方法調用n次

    Mockito.verify(userDAO, Mockito.times(n)).delete(userId);

    8.2.4.驗證方法調用至少1次

    Mockito.verify(userDAO, Mockito.atLeastOnce()).delete(userId);

    8.2.5.驗證方法調用至少n次

    Mockito.verify(userDAO, Mockito.atLeast(n)).delete(userId);

    8.2.2.驗證方法調用最多1次

    Mockito.verify(userDAO, Mockito.atMostOnce()).delete(userId);

    8.2.6.驗證方法調用最多n次

    Mockito.verify(userDAO, Mockito.atMost(n)).delete(userId);

    8.2.7.驗證方法調用指定n次

    Mockito允許按順序進行驗證方法調用,未被驗證到的方法調用將不會被標記為已驗證。

    Mockito.verify(userDAO, Mockito.call(n)).delete(userId);

    8.2.8.驗證對象及其方法調用1次

    用于驗證對象及其方法調用1次,如果該對象還有別的方法被調用或者該方法調用了多次,都將導致驗證方法調用失敗。

    Mockito.verify(userDAO, Mockito.only()).delete(userId);

    相當于:

    Mockito.verify(userDAO).delete(userId); Mockito.verifyNoMoreInteractions(userDAO);

    8.3.驗證方法調用并捕獲參數值

    Mockito提供ArgumentCaptor類來捕獲參數值,通過調用forClass(Class< T> clazz)方法來構建一個ArgumentCaptor對象,然后在驗證方法調用時來捕獲參數,最后獲取到捕獲的參數值并驗證。如果一個方法有多個參數都要捕獲并驗證,那就需要創建多個ArgumentCaptor對象。

    ArgumentCaptor的主要接口方法:

  • capture方法,用于捕獲方法參數;
  • getValue方法,用于獲取捕獲的參數值,如果捕獲了多個參數值,該方法只返回最后一個參數值;
  • getAllValues方法,用戶獲取捕獲的所有參數值。
  • 8.3.1.使用ArgumentCaptor.forClass方法定義參數捕獲器

    在測試用例方法中,直接使用ArgumentCaptor.forClass方法定義參數捕獲器。

    注意:定義泛型類的參數捕獲器時,存在強制類型轉化,會引起編譯器警告。

    8.3.2.使用@Captor注解定義參數捕獲器

    也可以用Mockito提供的@Captor注解,在測試用例類中定義參數捕獲器。

    @RunWith(PowerMockRunner.class) public class UserServiceTest {@Captorprivate ArgumentCaptor< UserDO> userCaptor;@Testpublic void testModifyUser() {...Mockito.verify(userDAO).modify(userCaptor.capture());UserDO user = userCaptor.getValue();} }

    注意:定義泛型類的參數捕獲器時,由于是Mockito自行初始化,不會引起編譯器告警。

    8.3.3.捕獲多次方法調用的參數值列表

    8.4.驗證其它特殊方法

    8.4.1.驗證final方法調用

    final方法的驗證跟普通方法類似,這里不再累述。

    8.4.2.驗證私有方法調用

    PowerMockito提供verifyPrivate方法驗證私有方法調用。

    PowerMockito.verifyPrivate(myClass, times(1)).invoke("unload", any(List.class));

    8.4.3.驗證構造方法調用

    PowerMockito提供verifyNew方法驗證構造方法調用。

    PowerMockito.verifyNew(MockClass.class).withNoArguments(); PowerMockito.verifyNew(MockClass.class).withArguments(someArgs);

    8.4.4.驗證靜態方法調用

    PowerMockito提供verifyStatic方法驗證靜態方法調用。

    PowerMockito.verifyStatic(StringUtils.class); StringUtils.isEmpty(string);

    9.驗證數據對象

    JUnit測試框架中Assert類就是斷言工具類,主要驗證單元測試中實際數據對象與期望數據對象一致。在調用被測方法時,需要對返回值和異常進行驗證;在驗證方法調用時,也需要對捕獲的參數值進行驗證。

    9.1.驗證數據對象空值

    9.1.1.驗證數據對象為空

    通過JUnit提供的Assert.assertNull方法驗證數據對象為空。

    Assert.assertNull("用戶標識必須為空", userId);

    9.1.2.驗證數據對象非空

    通過JUnit提供的Assert.assertNotNull方法驗證數據對象非空。

    Assert.assertNotNull("用戶標識不能為空", userId);

    9.2.驗證數據對象布爾值

    9.2.1.驗證數據對象為真

    通過JUnit提供的Assert.assertTrue方法驗證數據對象為真。

    Assert.assertTrue("返回值必須為真", NumberHelper.isPositive(1));

    9.2.2.驗證數據對象為假

    通過JUnit提供的Assert.assertFalse方法驗證數據對象為假。

    Assert.assertFalse("返回值必須為假", NumberHelper.isPositive(-1));

    9.3.驗證數據對象引用

    在單元測試用例中,對于一些參數或返回值對象,不需要驗證對象具體取值,只需要驗證對象引用是否一致。

    9.3.1.驗證數據對象一致

    JUnit提供的Assert.assertSame方法驗證數據對象一致。

    UserDO expectedUser = ...; Mockito.doReturn(expectedUser).when(userDAO).get(userId); UserDO actualUser = userService.getUser(userId); Assert.assertSame("用戶必須一致", expectedUser, actualUser);

    9.3.1.驗證數據對象不一致

    JUnit提供的Assert.assertNotSame方法驗證數據對象一致。

    UserDO expectedUser = ...; Mockito.doReturn(expectedUser).when(userDAO).get(userId); UserDO actualUser = userService.getUser(otherUserId); Assert.assertNotSame("用戶不能一致", expectedUser, actualUser);

    9.4.驗證數據對象值

    JUnit提供Assert.assertEquals、Assert.assertNotEquals、Assert.assertArrayEquals方法組,可以用來驗證數據對象值是否相等。

    9.4.1.驗證簡單數據對象

    對于簡單數據對象(比如:基礎類型、包裝類型、實現了equals的數據類型……),可以直接通過JUnit的Assert.assertEquals和Assert.assertNotEquals方法組進行驗證。

    Assert.assertNotEquals("用戶名稱不一致", "admin", userName); Assert.assertEquals("賬戶金額不一致", 10000.0D, accountAmount, 1E-6D);

    9.4.2.驗證簡單數組或集合對象

    對于簡單數組對象(比如:基礎類型、包裝類型、實現了equals的數據類型……),可以直接通過JUnit的Assert.assertArrayEquals方法組進行驗證。對于簡單集合對象,也可以通過Assert.assertEquals方法驗證。

    Long[] userIds = ...; Assert.assertArrayEquals("用戶標識列表不一致", new Long[] {1L, 2L, 3L}, userIds);List< Long> userIdList = ...; Assert.assertEquals("用戶標識列表不一致", Arrays.asList(1L, 2L, 3L), userIdList);

    9.4.3.驗證復雜數據對象

    對于復雜的JavaBean數據對象,需要驗證JavaBean數據對象的每一個屬性字段。

    UserDO user = ...; Assert.assertEquals("用戶標識不一致", Long.valueOf(1L), user.getId()); Assert.assertEquals("用戶名稱不一致", "admin", user.getName()); Assert.assertEquals("用戶公司標識不一致", Long.valueOf(1L), user.getCompany().getId()); ...

    9.4.4.驗證復雜數組或集合對象

    對于復雜的JavaBean數組和集合對象,需要先展開數組和集合對象中每一個JavaBean數據對象,然后驗證JavaBean數據對象的每一個屬性字段。

    List< UserDO> expectedUserList = ...; List< UserDO> actualUserList = ...; Assert.assertEquals("用戶列表長度不一致", expectedUserList.size(), actualUserList.size()); UserDO[] expectedUsers = expectedUserList.toArray(new UserDO[0]); UserDO[] actualUsers = actualUserList.toArray(new UserDO[0]); for (int i = 0; i < actualUsers.length; i++) {Assert.assertEquals(String.format("用戶(%s)標識不一致", i), expectedUsers[i].getId(), actualUsers[i].getId());Assert.assertEquals(String.format("用戶(%s)名稱不一致", i), expectedUsers[i].getName(), actualUsers[i].getName()); Assert.assertEquals("用戶公司標識不一致", expectedUsers[i].getCompany().getId(), actualUsers[i].getCompany().getId());... }

    9.4.5.通過序列化驗證數據對象

    如上一節例子所示,當數據對象過于復雜時,如果采用Assert.assertEquals依次驗證每個JavaBean對象、驗證每一個屬性字段,測試用例的代碼量將會非常龐大。這里,推薦使用序列化手段簡化數據對象的驗證,比如利用JSON.toJSONString方法把復雜的數據對象轉化為字符串,然后再使用Assert.assertEquals方法進行驗證字符串。但是,序列化值必須具備有序性、一致性和可讀性。

    List< UserDO> userList = ...; String text = ResourceHelper.getResourceAsString(getClass(), "userList.json"); Assert.assertEquals("用戶列表不一致", text, JSON.toJSONString(userList));

    通常使用JSON.toJSONString方法把Map對象轉化為字符串,其中key-value的順序具有不確定性,無法用于驗證兩個對象是否一致。這里,JSON提供序列化選項SerializerFeature.MapSortField(映射排序字段),可以用于保證序列化后的key-value的有序性。

    Map< Long, Map< String, Object>> userMap = ...; String text = ResourceHelper.getResourceAsString(getClass(), "userMap.json"); Assert.assertEquals("用戶映射不一致", text, JSON.toJSONString(userMap, SerializerFeature.MapSortField));

    9.4.6.驗證數據對象私有屬性字段

    有時候,單元測試用例需要對復雜對象的私有屬性字段進行驗證。而PowerMockito提供的Whitebox.getInternalState方法,獲取輕松地獲取到私有屬性字段值。

    MapperScannerConfigurer configurer = myBatisConfiguration.buildMapperScannerConfigurer(); Assert.assertEquals("基礎包不一致", "com.alibaba.example", Whitebox.getInternalState(configurer, "basePackage"));

    9.5.驗證異常對象內容

    異常作為Java語言的重要特性,是Java語言健壯性的重要體現。捕獲并驗證異常數據內容,也是測試用例的一種。

    9.5.1.通過@Test注解驗證異常對象

    JUnit的注解@Test提供了一個expected屬性,可以指定一個期望的異常類型,用來捕獲并驗證異常。但是,這種方式只能驗證異常類型,并不能驗證異常原因和消息。

    @Test(expected = ExampleException.class) public void testGetUser() {// 模擬依賴方法Mockito.doReturn(null).when(userDAO).get(userId);// 調用被測方法userService.getUser(userId); }

    9.5.2.通過@Rule注解驗證異常對象

    如果想要驗證異常原因和消息,就需求采用@Rule注解定義ExpectedException對象,然后在測試方法的前面聲明要捕獲的異常類型、原因和消息。

    @Rule private ExpectedException exception = ExpectedException.none(); @Test public void testGetUser() {// 模擬依賴方法Long userId = 123L;Mockito.doReturn(null).when(userDAO).get(userId);// 調用被測方法exception.expect(ExampleException.class);exception.expectMessage(String.format("用戶(%s)不存在", userId));userService.getUser(userId); }

    9.5.3.通過Assert.assertThrows驗證異常對象

    在最新版的JUnit中,提供了一個更為簡潔的異常驗證方式——Assert.assertThrows方法。

    @Test public void testGetUser() {// 模擬依賴方法Long userId = 123L;Mockito.doReturn(null).when(userDAO).get(userId);// 調用被測方法ExampleException exception = Assert.assertThrows("異常類型不一致", ExampleException.class, () -> userService.getUser(userId));Assert.assertEquals("異常消息不一致", "處理異常", exception.getMessage()); }

    10.驗證依賴對象

    10.1.驗證模擬對象沒有任何方法調用

    Mockito提供了verifyNoInteractions方法,可以驗證模擬對象在被測試方法中沒有任何調用。

    Mockito.verifyNoInteractions(idGenerator, userDAO);

    10.2.驗證模擬對象沒有更多方法調用

    Mockito提供了verifyNoMoreInteractions方法,在驗證模擬對象所有方法調用后使用,可以驗證模擬對象所有方法調用是否都得到驗證。如果模擬對象存在任何未驗證的方法調用,就會拋出NoInteractionsWanted異常。

    Mockito.verifyNoMoreInteractions(idGenerator, userDAO);

    備注:Mockito的verifyZeroInteractions方法與verifyNoMoreInteractions方法功能相同,但是目前前者已經被廢棄。

    10.3.清除模擬對象所有方法調用標記

    在編寫單元測試用例時,為了減少單元測試用例數和代碼量,可以把多組參數定義在同一個單元測試用例中,然后用for循環依次執行每一組參數的被測方法調用。為了避免上一次測試的方法調用影響下一次測試的方法調用驗證,最好使用Mockito提供clearInvocations方法清除上一次的方法調用。

    // 清除所有對象調用 Mockito.clearInvocations(); // 清除指定對象調用 Mockito.clearInvocations(idGenerator, userDAO);

    11.典型案例分析

    這里,只收集了幾個經典案例,解決了特定環境下的特定問題。

    11.1.測試框架特性導致問題

    在編寫單元測試用例時,或多或少會遇到一些問題,大多數是由于對測試框架特性不熟悉導致,比如:

  • Mockito不支持對靜態方法、構造方法、final方法、私有方法的模擬;
  • Mockito的any相關的參數匹配方法并不支持可空參數和空參數;
  • 采用Mockito的參數匹配方法時,其它參數不能直接用常量或變量,必須使用Mockito的eq方法;
  • 使用when-then語句模擬Spy對象方法會先執行真實方法,應該使用do-when語句;
  • PowerMock對靜態方法、構造方法、final方法、私有方法的模擬需要把對應的類添加到@PrepareForTest注解中;
  • PowerMock模擬JDK的靜態方法、構造方法、final方法、私有方法時,需要把使用這些方法的類加入到@PrepareForTest注解中,從而導致單元測試覆蓋率不被統計;
  • PowerMock使用自定義的類加載器來加載類,可能導致系統類加載器認為有類型轉化問題;需要加上@PowerMockIgnore({"javax.crypto.*"})注解,來告訴PowerMock這個包不要用PowerMock的類加載器加載,需要采用系統類加載器來加載。

    ……

  • 對于這些問題,可以根據提示信息查閱相關資料解決,這里就不再累述了。

    11.2.捕獲參數值已變更問題

    在編寫單元測試用例時,通常采用ArgumentCaptor進行參數捕獲,然后對參數對象值進行驗證。如果參數對象值沒有變更,這個步驟就沒有任何問題。但是,如果參數對象值在后續流程中發生變更,就會導致驗證參數值失敗。

    原始代碼:

    public < T> void readData(RecordReader recordReader, int batchSize, Function< Record, T> dataParser, Predicate< List<T>> dataStorage) {try {// 依次讀取數據Record record;boolean isContinue = true;List< T> dataList = new ArrayList<>(batchSize);while (Objects.nonNull(record = recordReader.read()) && isContinue) {// 解析添加數據T data = dataParser.apply(record);if (Objects.nonNull(data)) {dataList.add(data);}// 批量存儲數據if (dataList.size() == batchSize) {isContinue = dataStorage.test(dataList);dataList.clear();}}// 存儲剩余數據if (CollectionUtils.isNotEmpty(dataList)) {dataStorage.test(dataList);dataList.clear();}} catch (IOException e) {String message = READ_DATA_EXCEPTION;log.warn(message, e);throw new ExampleException(message, e);} }

    測試用例:

    @Test public void testReadData() throws Exception {// 模擬依賴方法// 模擬依賴方法: recordReader.readRecord record0 = Mockito.mock(Record.class);Record record1 = Mockito.mock(Record.class);Record record2 = Mockito.mock(Record.class);TunnelRecordReader recordReader = Mockito.mock(TunnelRecordReader.class);Mockito.doReturn(record0, record1, record2, null).when(recordReader).read();// 模擬依賴方法: dataParser.applyObject object0 = new Object();Object object1 = new Object();Object object2 = new Object();Function< Record, Object> dataParser = Mockito.mock(Function.class);Mockito.doReturn(object0).when(dataParser).apply(record0);Mockito.doReturn(object1).when(dataParser).apply(record1);Mockito.doReturn(object2).when(dataParser).apply(record2);// 模擬依賴方法: dataStorage.testPredicate< List< Object>> dataStorage = Mockito.mock(Predicate.class);Mockito.doReturn(true).when(dataStorage).test(Mockito.anyList());// 調用測試方法odpsService.readData(recordReader, 2, dataParser, dataStorage);// 驗證依賴方法// 模擬依賴方法: recordReader.readMockito.verify(recordReader, Mockito.times(4)).read();// 模擬依賴方法: dataParser.applyMockito.verify(dataParser, Mockito.times(3)).apply(Mockito.any(Record.class));// 驗證依賴方法: dataStorage.testArgumentCaptor< List< Object>> recordListCaptor = ArgumentCaptor.forClass(List.class);Mockito.verify(dataStorage, Mockito.times(2)).test(recordListCaptor.capture());Assert.assertEquals("數據列表不一致", Arrays.asList(Arrays.asList(object0, object1), Arrays.asList(object2)), recordListCaptor.getAllValues()); }

    問題現象:

    執行單元測試用例失敗,拋出以下異常信息:

    java.lang.AssertionError: 數據列表不一致 expected:<[[java.lang.Object@1e3469df, java.lang.Object@79499fa], [java.lang.Object@48531d5]]> but was:<[[], []]>

    問題原因:

    由于參數dataList在調用dataStorage.test方法后,都被主動調用dataList.clear方法進行清空。由于ArgumentCaptor捕獲的是對象引用,所以最后捕獲到了同一個空列表。

    解決方案:

    可以在模擬依賴方法dataStorage.test時,保存傳入參數的當前值進行驗證。代碼如下:

    11.3.模擬Lombok的log對象問題

    Lombok的@Slf4j注解,廣泛地應用于Java項目中。在某些代碼分支里,可能只有log記錄日志的操作,為了驗證這個分支邏輯被正確執行,需要在單元測試用例中對log記錄日志的操作進行驗證。

    原始方法:

    @Slf4j @Service public class ExampleService {public void recordLog(int code) {if (code == 1) {log.info("執行分支1");return;}if (code == 2) {log.info("執行分支2");return;}log.info("執行默認分支");}... }

    測試用例:

    @RunWith(PowerMockRunner.class) public class ExampleServiceTest {@Mockprivate Logger log;@InjectMocksprivate ExampleService exampleService;@Testpublic void testRecordLog1() {exampleService.recordLog(1);Mockito.verify(log).info("執行分支1");} }

    問題現象:

    執行單元測試用例失敗,拋出以下異常信息:

    Wanted but not invoked: logger.info("執行分支1");

    原因分析:

    經過調式跟蹤,發現ExampleService中的log對象并沒有被注入。通過編譯發現,Lombok的@Slf4j注解在ExampleService類中生成了一個靜態常量log,而@InjectMocks注解并不支持靜態常量的注入。

    解決方案:

    采用作者實現的FieldHelper.setStaticFinalField方法,可以實現對靜態常量的注入模擬對象。

    @RunWith(PowerMockRunner.class) public class ExampleServiceTest {@Mockprivate Logger log;@InjectMocksprivate ExampleService exampleService;@Beforepublic void beforeTest() throws Exception {FieldHelper.setStaticFinalField(ExampleService.class, "log", log);}@Testpublic void testRecordLog1() {exampleService.recordLog(1);Mockito.verify(log).info("執行分支1");} }

    11.4.兼容Pandora等容器問題

    阿里巴巴的很多中間件,都是基于Pandora容器的,在編寫單元測試用例時,可能會遇到一些坑。

    原始方法:

    @Slf4j public class MetaqMessageSender {@Autowiredprivate MetaProducer metaProducer;public String sendMetaqMessage(String topicName, String tagName, String messageKey, String messageBody) {try {// 組裝消息內容Message message = new Message();message.setTopic(topicName);message.setTags(tagName);message.setKeys(messageKey);message.setBody(messageBody.getBytes(StandardCharsets.UTF_8));// 發送消息請求SendResult sendResult = metaProducer.send(message);if (sendResult.getSendStatus() != SendStatus.SEND_OK) {String msg = String.format("發送標簽(%s)消息(%s)狀態錯誤(%s)", tagName, messageKey, sendResult.getSendStatus());log.warn(msg);throw new ReconsException(msg);}log.info(String.format("發送標簽(%s)消息(%s)狀態成功:%s", tagName, messageKey, sendResult.getMsgId()));// 返回消息標識return sendResult.getMsgId();} catch (MQClientException | RemotingException | MQBrokerException | InterruptedException e) {// 記錄消息異常Thread.currentThread().interrupt();String message = String.format("發送標簽(%s)消息(%s)狀態異常:%s", tagName, messageKey, e.getMessage());log.warn(message, e);throw new ReconsException(message, e);}} }

    測試用例:

    @RunWith(PowerMockRunner.class) public class MetaqMessageSenderTest {@Mockprivate MetaProducer metaProducer;@InjectMocksprivate MetaqMessageSender metaqMessageSender;@Testpublic void testSendMetaqMessage() throws Exception {// 模擬依賴方法SendResult sendResult = new SendResult();sendResult.setMsgId("msgId");sendResult.setSendStatus(SendStatus.SEND_OK);Mockito.doReturn(sendResult).when(metaProducer).send(Mockito.any(Message.class));// 調用測試方法String topicName = "topicName";String tagName = "tagName";String messageKey = "messageKey";String messageBody = "messageBody";String messageId = metaqMessageSender.sendMetaqMessage(topicName, tagName, messageKey, messageBody);Assert.assertEquals("messageId不一致", sendResult.getMsgId(), messageId);// 驗證依賴方法ArgumentCaptor< Message> messageCaptor = ArgumentCaptor.forClass(Message.class);Mockito.verify(metaProducer).send(messageCaptor.capture());Message message = messageCaptor.getValue();Assert.assertEquals("topicName不一致", topicName, message.getTopic());Assert.assertEquals("tagName不一致", tagName, message.getTags());Assert.assertEquals("messageKey不一致", messageKey, message.getKeys());Assert.assertEquals("messageBody不一致", messageBody, new String(message.getBody()));} }

    問題現象:

    執行單元測試用例失敗,拋出以下異常信息:

    java.lang.RuntimeException: com.alibaba.rocketmq.client.producer.SendResult was loaded by org.powermock.core.classloader.javassist.JavassistMockClassLoader@5d43661b, it should be loaded by Pandora Container. Can not load this fake sdk class.

    原因分析:

    基于Pandora容器的中間件,需要使用Pandora容器加載。在上面測試用例中,使用了PowerMock容器加載,從而導致拋出類加載異常。

    解決方案:

    首先,把PowerMockRunner替換為PandoraBootRunner。其次,為了使@Mock、@InjectMocks等Mockito注解生效,需要調用MockitoAnnotations.initMocks(this)方法進行初始化。

    @RunWith(PandoraBootRunner.class) public class MetaqMessageSenderTest {...@Beforepublic void beforeTest() {MockitoAnnotations.initMocks(this);}... }

    12.消除類型轉換警告

    在編寫測試用例時,特別是泛型類型轉換時,很容易產生類型轉換警告。常見類型轉換警告如下:

    作為一個有代碼潔癖的輕微強迫癥程序員,是絕對不容許這些類型轉換警告產生的。于是,總結了以下方法來解決這些類型轉換警告。

    12.1.利用注解初始化

    Mockito提供@Mock注解來模擬類實例,提供@Captor注解來初始化參數捕獲器。由于這些注解實例是通過測試框架進行初始化的,所以不會產生類型轉換警告。

    問題代碼:

    建議代碼:

    12.2.利用臨時類或接口

    我們無法獲取泛型類或接口的class實例,但是很容易獲取具體類的class實例。這個解決方案的思路是——先定義繼承泛型類的具體子類,然后mock、spy、forClass以及any出這個具體子類的實例,然后把具體子類實例轉換為父類泛型實例。

    問題代碼:

    建議代碼:

    12.3.利用CastUtils.cast方法

    SpringData包中提供一個CastUtils.cast方法,可以用于類型的強制轉換。這個解決方案的思路是——利用CastUtils.cast方法屏蔽類型轉換警告。

    問題代碼:

    建議代碼:

    這個解決方案,不需要定義注解,也不需要定義臨時類或接口,能夠讓測試用例代碼更為精簡,所以作者重點推薦。如果不愿意引入SpringData包,也可以自己參考實現該方法,只是該方法會產生類型轉換警告。

    注意:CastUtils.cast方法本質是——先轉換為Object類型,再強制轉換對應類型,本身不會對類型進行校驗。所以,CastUtils.cast方法好用,但是不要亂用,否則就是大坑(只有執行時才能發現問題)。

    12.4.利用類型自動轉換

    在Mockito中,提供形式如下的方法——泛型類型只跟返回值有關,而跟輸入參數無關。這樣的方法,可以根據調用方法的參數類型自動轉換,而無需手動強制類型轉換。如果手動強制類型轉換,反而會產生類型轉換警告。

    問題代碼:

    建議代碼:

    其實,SpringData的CastUtils.cast方法之所以這么強悍,也是采用了類型自動轉化方法。

    12.5.利用doReturn-when語句代替when-thenReturn語句

    Mockito的when-thenReturn語句需要對返回類型強制校驗,而doReturn-when語句不會對返回類型強制校驗。利用這個特性,可以利用doReturn-when語句代替when-thenReturn語句解決類型轉換警告。

    問題代碼:

    建議代碼:

    12.6.利用instanceof關鍵字

    JDK提供的Method.invoke方法返回的是Object類型,轉化為具體類型時需要強制轉換,會產生類型轉換警告。而PowerMock提供的Whitebox.invokeMethod方法返回類型可以自動轉化,不會產生類型轉換警告。

    問題代碼:

    建議代碼:

    12.7.利用instanceof關鍵字

    在具體類型強制轉換時,建議利用instanceof關鍵字先判斷類型,否則會產生類型轉換警告。

    JSONArray jsonArray = (JSONArray)object; ...

    建議代碼:

    if (object instanceof JSONArray) {JSONArray jsonArray = (JSONArray)object;... }

    12.8.利用Class.cast方法

    在泛型類型強制轉換時,會產生類型轉換警告。可以采用泛型類的cast方法轉換,從而避免產生類型轉換警告。

    問題代碼:

    建議代碼:

    12.9.避免不必要的類型轉換

    有時候,沒有必要進行類型轉換,就盡量避免類型轉換。比如:把Object類型轉換為具體類型,但又把具體類型當Object類型使用,就沒有必要進行類型轉換。像這種情況,可以合并表達式或定義基類變量,從而避免不必要的類型轉化。

    問題代碼:

    建議代碼:

    后記

    登妙峰山記
    山高路遠車難騎,
    精疲力盡人易棄。
    多少妙峰登頂者,
    又練心境又練力!

    騎行的人,一定要沉得住氣、要吃得了苦、要耐得住寂寞、要意志堅定不移、要體力夠猛夠持久……恰好,這也正是技術人所要具備的精神。只要技術人做到了這些,練就了好的“心境”和“體力”,才有可能登上技術的“妙峰山”。

    原文鏈接

    本文為阿里云原創內容,未經允許不得轉載。

    總結

    以上是生活随笔為你收集整理的Java编程技巧之单元测试用例编写流程的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

    亚洲一区久久 | 久久综合免费 | 日本久久中文字幕 | 五月天视频网 | 就要色综合 | a√天堂中文在线 | 超碰精品在线观看 | 国产精品美女视频 | 国产精品麻豆三级一区视频 | 亚洲国产美女精品久久久久∴ | 久久久久久久久影院 | 欧美日韩国产高清视频 | 99免费在线观看 | 狠狠色丁香久久婷婷综合_中 | 国产福利91精品一区二区三区 | 亚洲在线视频免费观看 | 成 人 免费 黄 色 视频 | 亚洲色影爱久久精品 | 天天操天天干天天玩 | 欧美在线观看视频一区二区三区 | 精品九九九九 | 国产福利av在线 | 最近中文字幕大全 | 在线观看中文字幕2021 | 久久国产精品99精国产 | 视频一区二区精品 | 久久免费在线观看视频 | 国内精品视频在线播放 | 日日爱999 | 最近免费在线观看 | 亚洲精品国精品久久99热 | 午夜免费在线观看 | a久久久久久| a√资源在线 | 四虎天堂 | 精品产品国产在线不卡 | 天天干天天摸天天操 | 久久久久电影网站 | 亚洲精品国产拍在线 | 亚洲精品免费在线播放 | 日本在线h | 欧美日韩午夜爽爽 | 999免费视频| 精品久久久久久久久中文字幕 | 91精品国产欧美一区二区 | 探花视频在线观看+在线播放 | 精品国产美女在线 | 亚洲精品一区二区精华 | 九九色网 | 91精品久久久久久久久久入口 | 亚洲精品在线播放视频 | 成人免费在线视频观看 | 国产一区视频在线 | 久久在线一区 | 日韩电影在线视频 | 久久只精品99品免费久23小说 | 97色在线 | 99精品视频在线观看 | 免费又黄又爽 | 亚洲激情电影在线 | 久久久一本精品99久久精品 | 一区 二区 精品 | 亚洲国产日韩欧美 | 日韩成人av在线 | www.久久久 | 亚洲一区二区三区在线看 | 81国产精品久久久久久久久久 | 日韩免费av在线 | 国产精品九九九九九九 | 揉bbb玩bbb少妇bbb | 国产精品综合久久久久 | 国产精品美女视频 | 91免费观看国产 | 日韩视频免费在线 | 久久国产精品免费观看 | 91av官网 | 天天综合色 | 国产美女精品视频 | 国产在线理论片 | 久久爱资源网 | 人人爽人人片 | 高清中文字幕av | av免费在线观看1 | 天天操天天摸天天射 | 国产精品久久久久久久久久99 | 日韩日韩日韩日韩 | 九九免费在线观看 | 在线看v片 | 黄色一级大片免费看 | 久久视频国产精品免费视频在线 | 国产福利91精品一区二区三区 | 欧美成年人在线视频 | 免费看黄的 | 视频一区二区免费 | 国产精品久久一 | h网站免费在线观看 | 午夜久久福利影院 | 黄色aaaaa| 久久这里只有精品久久 | 99 精品 在线| 国产 av 日韩| 日本中文字幕系列 | 国产黄色大全 | 久久久一本精品99久久精品66 | 欧美久草视频 | 91社区国产高清 | 狠狠干网 | 久久久久久久久久电影 | 区一区二在线 | 亚洲精品高清视频在线观看 | 亚洲国产精品激情在线观看 | 97看片网 | 国产视频第二页 | 又黄又刺激的网站 | 免费99视频 | 亚洲最新在线视频 | 91完整版在线观看 | 粉嫩av一区二区三区免费 | 亚洲欧美日韩一二三区 | 91中文字幕在线视频 | 五月婷久久 | 国内成人精品视频 | 欧美亚洲国产一卡 | 日韩欧美xxx | 成人资源网 | 免费亚洲一区二区 | 亚洲精品午夜aaa久久久 | 中文字幕日韩av | 久久精品视频在线播放 | 一区二区三区四区在线 | 国产黄网站在线观看 | 久草国产在线 | 欧美日本不卡 | 久久国产精品99久久久久久老狼 | 狠狠干综合网 | 欧美日韩高清免费 | 国产99色 | 三级a毛片 | 五月天久久精品 | 国产精品手机看片 | 99热这里只有精品免费 | 欧美性久久久久久 | 四虎在线视频免费观看 | 欧洲高潮三级做爰 | 国产精品毛片一区二区 | 国产精品久久久久久久av大片 | 欧美激情精品久久久 | 婷婷深爱 | 二区三区精品 | 摸bbb搡bbb搡bbbb| 国产精品色 | 99精品国产成人一区二区 | 麻豆一区二区 | 国产精品久久久久久久久久不蜜月 | 91精品国产三级a在线观看 | 国产精品久久久久久久久久久久冷 | 久久五月婷婷丁香社区 | 午夜精品一区二区三区四区 | 国产成人精品一区二区三区福利 | 99久久影院| 一区二区精品在线 | 欧美日韩三区二区 | 亚洲一级黄色片 | 天天干婷婷| 2023av在线| 又色又爽又黄 | 伊人中文字幕在线 | 欧美伦理电影一区二区 | 欧美一级视频在线观看 | 五月婷婷丁香色 | 亚洲最新av在线网址 | 国产精品video爽爽爽爽 | 久久精品中文字幕一区二区三区 | 中文字幕麻豆 | 国产免费又爽又刺激在线观看 | 99久热在线精品视频观看 | 亚洲欧美va| 日韩在线第一 | 日韩在线视频免费看 | 亚洲人成在线观看 | 四虎永久精品在线 | 国产精品手机在线观看 | 久久久久久久99精品免费观看 | 在线观看岛国片 | 黄色aaa级片 | 婷婷丁香狠狠爱 | 久久av免费电影 | 成年人免费在线播放 | 中文字幕在线观看播放 | 国产精品一区久久久久 | 亚洲成人av在线播放 | 欧美日韩伦理一区 | 美女免费视频一区二区 | 日韩久久影院 | 欧美国产日韩在线视频 | 999久久 | 天天天天天天操 | 91污视频在线观看 | 亚洲综合色站 | 国产精品美女久久久久久久网站 | 超碰午夜 | 久久99久久久久 | 欧美视频二区 | 国产精品麻豆果冻传媒在线播放 | 久久丁香网 | 黄色的视频 | 91成人黄色 | 色99在线| 婷婷丁香久久五月婷婷 | japanese黑人亚洲人4k | www.久久爱.cn | 亚洲人成影院在线 | 国产亚洲人成网站在线观看 | 蜜桃av久久久亚洲精品 | 激情综合网色播五月 | 午夜影院一级 | 久久免费在线观看 | 色婷婷狠| 亚洲黄色av网址 | 天天干天天看 | 久久九九久久精品 | 国产高清视频免费最新在线 | 亚洲精品高清视频 | 国产91av视频在线观看 | 久久久久亚洲精品成人网小说 | 激情欧美xxxx | 婷婷久久网 | 国产成人综合在线观看 | 亚洲激情视频在线观看 | 在线观看中文av | 亚洲v欧美v国产v在线观看 | 91免费视频网站在线观看 | 日韩精品亚洲专区在线观看 | 成年人在线视频观看 | 国产成人黄色片 | 又黄又爽的免费高潮视频 | av电影中文字幕 | 深夜免费福利 | 国产精品成久久久久三级 | 日韩在线在线 | 五月天激情开心 | 天天综合视频在线观看 | 精品国产一区二区三区久久久蜜臀 | 国产黄色在线网站 | 国产蜜臀av | 狠狠色狠狠色综合日日92 | 国产成人性色生活片 | 亚洲国产精品小视频 | 美女网站免费福利视频 | 一区二区三区电影在线播 | 黄色精品免费 | 三级视频国产 | 国产不卡在线观看 | 男女精品久久 | 日本视频网 | 欧美福利视频一区 | 亚洲精品福利在线观看 | 久久久999免费视频 日韩网站在线 | 91av视频播放 | 97超碰人人澡| 五月天久久久久久 | 精品免费久久 | 日韩超碰在线 | 制服丝袜亚洲 | 在线亚洲精品 | 91热爆视频 | 久久精品美女视频网站 | www.超碰| 一级片免费观看视频 | 久草在线一免费新视频 | 天天干天天做 | 免费69视频 | 国内亚洲精品 | 一区二区三区四区五区在线 | 久久99热这里只有精品 | 日本护士三级少妇三级999 | 亚洲高清av | 亚洲女欲精品久久久久久久18 | 久久综合中文色婷婷 | 久久8| 18pao国产成视频永久免费 | www.91成人 | 日本久久精 | 国色天香av | 久草在线综合 | 黄色亚洲免费 | 久久视频在线观看中文字幕 | 中文字幕在线看片 | 91亚洲精品久久久蜜桃借种 | 久久伦理电影 | 波多野结衣小视频 | 天天干天天射天天插 | 国产黄色网 | 中文字幕在线字幕中文 | 美女久久久久久久久久久 | 欧美精品首页 | 在线精品播放 | 一性一交视频 | 国产91aaa | 成人久久精品视频 | 婷婷久操 | 色网站在线 | 国产成人精品福利 | 国产亚洲一级高清 | 一区二区三区在线免费播放 | 黄a在线看| 久久精品综合 | 亚洲精品毛片一级91精品 | 欧美在线视频不卡 | 国产资源 | 国产99久久久国产精品成人免费 | 午夜久久影视 | 国产精品久久久久久99 | 超碰资源在线 | 波多野结衣在线观看一区二区三区 | 久久蜜桃av| 亚洲欧美视频在线观看 | 午夜久久久久久久久 | 国产成人一区二区三区在线观看 | 国产精品美女999 | 午夜精品福利一区二区 | 欧美日韩免费观看一区=区三区 | 成人中文字幕+乱码+中文字幕 | 久久综合色综合88 | 精壮的侍卫呻吟h | 精品国产乱码久久久久久1区二区 | 欧美久久成人 | 欧美美女视频在线观看 | 麻豆国产视频下载 | 免费av视屏 | 99精品国产99久久久久久福利 | 国产私拍在线 | 天天干夜夜操视频 | 99在线观看 | 亚洲国产精品va在线看黑人 | 国产精品18久久久久vr手机版特色 | 九九在线视频免费观看 | 国产中文字幕视频在线观看 | 日韩av影视在线 | 91久久精品日日躁夜夜躁国产 | 久久激情小说 | 久久在线精品视频 | 午夜电影久久久 | 成人黄色片免费看 | 欧洲高潮三级做爰 | 天天草天天干天天 | 久久69精品久久久久久久电影好 | 欧美a级免费视频 | 在线观看片 | 久久99精品国产一区二区三区 | 亚洲一区天堂 | 国产高清免费观看 | av成人在线网站 | 四虎影院在线观看av | 一区二区三区免费播放 | 波多野结衣一区 | 久久99国产精品自在自在app | 日韩精品一区二区三区免费观看视频 | 久久久国产毛片 | 亚洲视频资源在线 | 色丁香色婷婷 | 97人人艹| 99精品视频免费观看视频 | 久久天堂影院 | 久久久久久免费网 | 国产精品成人一区二区三区 | 欧美9999 | 国产一区二区免费看 | 日韩av一区二区在线播放 | 日韩欧美在线观看一区二区 | 国产原创av在线 | 日韩免费一级电影 | 中文字幕av日韩 | 91精品国自产在线观看欧美 | 久久精品一区二区 | 亚洲三级在线 | 亚洲午夜大片 | 久久亚洲精品电影 | 国产视频精品在线 | 国产成人免费网站 | 国产男男gay做爰 | 国产亲近乱来精品 | 欧美最猛性xxx| 国产99久久久久 | 99热99| 日本黄色免费网站 | 日韩乱色精品一区二区 | 成人av影视| 黄色91在线 | 日韩在观看线 | 久久一线 | 韩国av三级 | 狠色在线 | 亚洲女人天堂成人av在线 | 最近中文字幕免费av | 999免费视频| 亚洲精品久久久久久久不卡四虎 | 亚洲一区二区高潮无套美女 | 亚洲一区天堂 | 免费av的网站| 国产91在线观看 | 美女久久久久久久久久 | 中文字幕丝袜制服 | 91麻豆文化传媒在线观看 | zzijzzij日本成熟少妇 | 狠狠干网站 | 亚洲午夜精 | 丝袜美腿av| 欧美日一级片 | 精品美女久久 | 成人18视频| 欧美一区影院 | 中文字幕在线观看免费观看 | 麻豆国产网站入口 | 国产欧美日韩精品一区二区免费 | 一区二区国产精品 | 欧美一区二区三区激情视频 | 精品视频久久久 | 97福利| 国产手机视频在线 | 免费日韩一区 | 欧美一区三区四区 | 手机在线日韩视频 | 在线观看免费成人 | 国产精品美女网站 | 99国产视频 | 九七视频在线观看 | 激情久久久久久久久久久久久久久久 | 日韩欧美视频一区二区 | 国产精品久久久久一区二区三区共 | 在线激情影院一区 | 一本一本久久a久久 | 在线观看日韩中文字幕 | 久久视频免费在线 | 欧美一区二区在线刺激视频 | 五月天婷亚洲天综合网精品偷 | 国产原创中文在线 | 欧美日韩在线观看不卡 | 一本一道久久a久久精品蜜桃 | 国产精品国产三级国产 | 日韩中文字幕国产 | 久久激情五月婷婷 | 色综合久久久 | 日韩在线观看中文字幕 | 蜜臀一区二区三区精品免费视频 | 黄色午夜网站 | 国产高清不卡一区二区三区 | 夜色在线资源 | 欧美日韩亚洲国产一区 | 午夜精品999 | 天天舔天天射天天操 | 中文字幕一区二区三区在线观看 | 丁香狠狠 | 亚洲电影网站 | 成人国产精品一区 | www.久久久精品 | 色婷婷久久一区二区 | 日韩成人在线一区二区 | 超碰在线公开 | 99久久久国产精品 | 国产一区二区三区黄 | 欧美一区二区三区四区夜夜大片 | 日韩午夜网站 | 国色天香av | 亚洲va欧美 | 欧美成人精品在线 | 国产精品久久久久国产a级 激情综合中文娱乐网 | 久久www免费人成看片高清 | 欧美不卡视频在线 | 日韩黄色在线 | www.久久色| 国产欧美综合在线观看 | 国产视频2区 | 午夜精品导航 | 国产一区二区在线免费播放 | 国产精品久久久久久久免费观看 | www.国产高清 | 91在线免费视频观看 | 欧美视频日韩 | 91.麻豆视频| 久久精品视频网站 | 2000xxx影视 | 国产精品麻豆一区二区三区 | 国产精品免费视频一区二区 | 久久色亚洲 | 亚州av免费 | 亚洲成人精品久久 | 在线观看中文av | 国产在线色 | 久久女同性恋中文字幕 | 国产91综合一区在线观看 | 91手机在线看片 | 91在线成人 | 亚洲高清av| 伊人久在线 | 国产精选在线 | 午夜av在线电影 | 国产色女| 免费看的黄色的网站 | 亚洲精品美女免费 | 狠狠操狠狠干天天操 | 国产黄色av影视 | 国产精品一区二区吃奶在线观看 | 中文字幕av免费观看 | a视频在线观看免费 | 欧美大荫蒂xxx | 国产一级电影网 | 欧美精品视 | 天天干天天操天天搞 | 久久久www成人免费毛片麻豆 | 久久8| 国产精美视频 | 中文av在线天堂 | 69国产精品成人在线播放 | 国产视频一区二区在线观看 | 国产又粗又猛又色又黄视频 | 丁香六月婷婷开心婷婷网 | 成人国产精品免费观看 | 91精品视频免费看 | 日韩精品中文字幕av | 九九久久精品视频 | 久久综合九色综合欧美就去吻 | 免费在线观看国产精品 | 欧美成人一区二区 | 欧美日韩精品在线观看 | 日韩精品免费一线在线观看 | 黄色av一级 | 一区二区中文字幕在线观看 | 国产色综合 | 日韩av三区 | 日本中文字幕在线免费观看 | 在线免费中文字幕 | 成年人视频在线观看免费 | 日韩av免费一区二区 | 日韩乱理 | 久久久久久久福利 | 亚洲综合成人婷婷小说 | 日韩精品不卡在线 | ,午夜性刺激免费看视频 | 亚洲精品中文字幕在线 | 丁香五香天综合情 | 天天躁日日躁狠狠躁av麻豆 | 国产在线传媒 | 久久久久成人精品亚洲国产 | av手机版 | 999免费视频 | 激情电影在线观看 | 日日干综合 | 天天干一干 | 久久精品看片 | 国产一区二区三区免费在线 | 日韩高清精品一区二区 | a在线免费观看视频 | 欧美激情视频一区二区三区 | 久久五月婷婷综合 | 最近更新中文字幕 | 亚洲成人av在线 | 国产免费二区 | 亚洲永久精品视频 | 久草视频99 | 日韩欧美精品在线观看视频 | 久久久久久久久综合 | 久久爱综合 | 日韩精品久久久久久久电影竹菊 | 97在线免费视频 | 国产精品免费大片视频 | 久久久蜜桃一区二区 | 亚洲综合欧美激情 | 日本久久免费电影 | 国产精品爽爽久久久久久蜜臀 | 欧美五月婷婷 | 九九热精品视频在线观看 | 91中文字幕视频 | 国产精品一区二区三区电影 | av一本久道久久波多野结衣 | www.狠狠操 | 国产网红在线观看 | 亚一亚二国产专区 | 91一区啪爱嗯打偷拍欧美 | 久久黄页 | 综合色婷婷 | 少妇搡bbb | 婷婷.com| 免费福利在线观看 | 成年美女黄网站色大片免费看 | 国产高清在线不卡 | 999久久 | 欧美国产日韩一区二区 | 欧美日韩视频观看 | 日韩精品第一区 | 国产一区黄色 | 99热国产在线 | 九九视频在线 | 国产精品成人一区二区三区吃奶 | 黄色av电影在线 | 午夜18视频在线观看 | 婷婷亚洲最大 | 久久国产精品久久精品国产演员表 | 国产不卡视频在线 | 久久久久久久久久久高潮一区二区 | av福利超碰网站 | 国产精品一区二区你懂的 | 国产高清在线a视频大全 | 国产一区视频在线播放 | 日韩二区三区在线 | 国产精品日韩高清 | 欧美日韩国产页 | 狠狠色丁香婷婷综合基地 | 五月天色中色 | a久久免费视频 | 国产无遮挡猛进猛出免费软件 | 欧美专区国产专区 | 天天摸天天舔天天操 | 97超碰人人在线 | 亚洲精品美女久久久久 | 在线观看国产一区二区 | 伊人婷婷久久 | 人人澡视频| 在线观看av中文字幕 | 国产精品va在线观看入 | 丝袜一区在线 | 丁香六月久久综合狠狠色 | 国产麻豆剧果冻传媒视频播放量 | 中文字幕av最新更新 | 91精品在线免费观看视频 | 成人免费视频免费观看 | 国产小视频91 | 久久精品视频在线免费观看 | 成人免费一区二区三区在线观看 | 好看的国产精品视频 | 五月开心婷婷网 | 国产99在线播放 | 久久人人添人人爽添人人88v | 美女视频a美女大全免费下载蜜臀 | 中文字幕乱码亚洲精品一区 | 香蕉在线观看 | 日本三级香港三级人妇99 | 亚洲国产精品推荐 | 日韩无在线 | 韩国av免费观看 | 亚洲精品在线免费播放 | 91九色porny蝌蚪视频 | 精品亚洲网 | 五月天视频网站 | 免费看三级黄色片 | 中文字幕韩在线第一页 | 蜜桃av久久久亚洲精品 | av免费福利 | 国产精品成人一区二区三区 | 国产精品久久久久影院 | 人人爽人人 | 日本中文字幕在线免费观看 | 免费在线观看成人av | 激情黄色av| 日韩亚洲在线观看 | 五月av在线| 国产成人在线免费观看 | 在线视频1卡二卡三卡 | 欧美精品被 | 亚洲成人av一区 | 91热这里只有精品 | 亚洲黄在线观看 | 中文字幕精品一区久久久久 | 久久精品黄色 | 国产色视频| 欧美精品乱码久久久久 | 亚洲激情视频在线观看 | 奇米7777狠狠狠琪琪视频 | 国产精品乱码一区二区视频 | 麻豆视频免费看 | 婷婷丁香在线 | 一级黄色视屏 | 久久国产精品免费一区二区三区 | 97**国产露脸精品国产 | 丁香五月缴情综合网 | 欧美午夜精品久久久久久浪潮 | 中文字幕在线国产 | 波多野结衣在线观看一区 | 亚洲视频在线免费看 | 精品视频免费看 | 日韩免费av片 | 黄色在线看网站 | 日韩精品无 | 婷婷丁香国产 | 国产精品久久网 | 一区二区三区日韩精品 | 久久综合九色99 | 五月激情五月激情 | 国产91九色视频 | 亚洲影院色 | 日韩精品一区二区三区中文字幕 | 91av美女| 超碰97在线资源站 | 91毛片在线 | 99久久www| 日韩国产欧美在线播放 | av网址在线播放 | 伊人久久一区 | 久久综合99| 91麻豆精品久久久久久 | av蜜桃在线 | 亚洲精品乱码久久久久久久久久 | 人人揉人人揉人人揉人人揉97 | 亚洲桃花综合 | 一区二区三区视频网站 | 九九热精品视频在线播放 | 亚洲精品网址在线观看 | 91av原创| 国产精品成人av久久 | 国产精品区免费视频 | 精品国产一区二区三区四 | 最近免费中文字幕 | 岛国av在线免费 | 香蕉视频在线播放 | 午夜免费电影院 | 97视频入口免费观看 | 人人超碰免费 | 91九色在线播放 | 97色在线观看免费视频 | 97超碰人人模人人人爽人人爱 | 国产精品毛片一区二区在线 | 欧美一级片免费观看 | 超碰公开在线 | 麻豆影视在线观看 | 天天噜天天色 | 日韩免费一级电影 | 亚洲视频每日更新 | 欧美一区二区在线免费看 | 亚洲一级二级 | 超碰97人人爱 | 天天干天天在线 | 丰满少妇在线观看资源站 | 婷婷色影院 | 国产成人精品在线播放 | 免费亚洲电影 | 国产精品女同一区二区三区久久夜 | 狠狠干婷婷色 | 中文字幕国内精品 | 97综合在线| 久草视频免费 | 久久99国产精品久久99 | 午夜精品一二三区 | 成年人天堂com| 草久中文字幕 | 日韩资源视频 | 久久歪歪| 午夜美女福利直播 | 婷婷色视频| 三级黄色在线 | 国产一区二区精品久久91 | 免费看一级特黄a大片 | 精品福利在线视频 | 亚洲成a人片综合在线 | 国产成人精品久久 | 婷婷伊人综合亚洲综合网 | 色婷婷啪啪免费在线电影观看 | 激情五月在线视频 | 天天操天天干天天摸 | 97超级碰 | 91久久国产综合精品女同国语 | 新av在线 | 欧美日本三级 | 成人在线视频你懂的 | 国产亚洲资源 | 亚洲第一色 | 国产精品久久99综合免费观看尤物 | 在线免费av播放 | 在线观看视频免费大全 | 亚洲精品中文在线 | 免费观看xxxx9999片 | 在线之家免费在线观看电影 | 天天操夜操 | 日韩在线观看小视频 | 欧美性生爱 | 黄色软件在线观看视频 | 91av视频播放 | 激情综合啪| 日韩在线精品一区 | 国产资源免费在线观看 | 欧美淫视频 | 国产成人高清在线 | 福利二区视频 | 一区二区电影在线观看 | 国产精品久久久久亚洲影视 | 女人18片 | 色欧美88888久久久久久影院 | 亚洲欧美日本A∨在线观看 青青河边草观看完整版高清 | 欧美精品午夜 | 国产色视频一区二区三区qq号 | 国产精品午夜免费福利视频 | 欧美成年人在线视频 | 久久久国产精品一区二区三区 | 97精品国产97久久久久久粉红 | www.天天操.com| aaa日本高清在线播放免费观看 | 日韩精品一区二区三区高清免费 | 91在线精品一区二区 | 国产成人亚洲在线观看 | 在线观看免费av网站 | 欧美在线你懂的 | 女人18片| 一级成人免费视频 | 国产精品久久久久久久久久了 | 在线观看免费成人 | 免费成人黄色片 | 久久综合欧美 | 国产中文在线观看 | av大全免费在线观看 | 久久婷婷精品视频 | av高清免费 | 中文字幕精品一区久久久久 | 极品久久久 | 天天天天综合 | 在线看一区二区 | 婷婷黄色片 | 西西大胆啪啪 | 最新一区二区三区 | 久久avav | 在线视频日韩一区 | 国产免费一区二区三区最新6 | 国产精品久久久久久久久岛 | 精品字幕在线 | 最新的av网站 | 国产成人精品亚洲日本在线观看 | 丁香花在线观看视频在线 | 久久永久免费 | 免费成人在线网站 | 大胆欧美gogo免费视频一二区 | 亚洲最大成人免费网站 | 天天摸天天操天天舔 | 国产成人av一区二区三区在线观看 | 久草在线资源观看 | 黄色片视频在线观看 | 一区二区欧美在线观看 | 亚洲精品啊啊啊 | 天天摸夜夜添 | 丁香六月av | 亚洲va欧美va人人爽春色影视 | 国产亚洲综合在线 | 久久呀 | 亚洲精品国产区 | 国产精品国内免费一区二区三区 | 久久超碰在线 | 狠狠躁日日躁夜夜躁av | 91干干干| 18国产精品福利片久久婷 | 日韩精品久久久久久久电影99爱 | 国内精品久久久久影院一蜜桃 | 四虎永久免费 | 欧美大码xxxx | 亚洲h视频在线 | 久艹在线免费观看 | 麻豆91网站 | 亚洲国产人午在线一二区 | 免费人成在线观看网站 | 日韩精品大片 | 国产一区二三区好的 | 免费色视频网址 | 99r精品视频在线观看 | 国产午夜一区 | 五月激情在线 | 久久99精品久久久久久久久久久久 | a成人v在线 | 日韩xxxx视频 | 久久久久亚洲精品成人网小说 | 亚洲高清色综合 | 国产精品久久久久久久久久久久午夜 | 久久国产乱 | 国产成人精品女人久久久 | 国产精品电影一区 | 免费av影视 | 免费视频二区 | 久久福利| 国产69久久精品成人看 | 亚洲涩涩涩涩涩涩 | 热久久视久久精品18亚洲精品 | 日韩手机在线观看 | 超碰在线中文字幕 | 综合色中文 | 亚洲午夜精品久久久久久久久久久久 | 亚洲视频网站在线观看 | 97av视频| 色黄视频免费观看 | 亚洲婷婷在线视频 | 久久久久国 | 免费看成人 | 久久精品综合视频 | 999久久久欧美日韩黑人 | 韩日精品中文字幕 | 日韩免费观看av | 国产欧美久久久精品影院 | 久久99久久99免费视频 | av中文字幕免费在线观看 | 欧美一区二区日韩一区二区 | 不卡av在线 | 国产日韩欧美视频 | 91视频xxxx| 九九在线精品视频 | 日韩久久网站 | 91视频成人免费 | 人人草在线观看 | 91精品对白一区国产伦 | 91福利专区 | 伊甸园永久入口www 99热 精品在线 | 色无五月| 色网站在线看 | 99亚洲国产精品 | 激情婷婷亚洲 | 亚洲精品xx| 五月宗合网 | 欧美日韩视频在线 | 国产黄色大片 | 久久伊人精品天天 | 一区二区伦理电影 | 超级av在线 | 欧美视频www| 国产成人三级 | 亚洲更新最快 | 日日麻批40分钟视频免费观看 | 激情五月在线 | 亚洲成人精品影院 | 精品一区二区在线播放 | 丁香婷五月 | 超级碰碰碰免费视频 | 中文字幕在线观看免费高清电影 | 久久精品99国产国产精 | 麻豆播放 | 欧美中文字幕久久 | 中文字幕888 | 日黄网站 | 在线影院av | 国产精品视频你懂的 | 日批视频在线 | 国产成人一区二区精品非洲 | a黄色一级片 | 久久99久久99精品免费看小说 | 2019天天干夜夜操 | 中文字幕免费久久 | 国产日韩中文在线 | 国产成人一区二区三区影院在线 | 日韩激情免费视频 | 成年人在线免费视频观看 | 国产在线观看高清视频 | 亚洲久在线 | 最新国产在线视频 | 国产麻豆精品久久 | 欧美夫妻性生活电影 | 天天操狠狠操 | 97超级碰碰碰碰久久久久 | 久久亚洲欧美 | 免费在线观看日韩视频 | 久久精品这里都是精品 | 91在线免费观看网站 | 91视频链接 | aa一级片 | av一本久道久久波多野结衣 | 2018亚洲男人天堂 | 久久久国产视频 | 亚洲精品字幕在线观看 | 香蕉视频4aa | 久久视| 人人爽人人爽人人片av免 | 黄色影院在线免费观看 | 免费三级av | 国产精品久久久久久99 | 黄色一级片视频 | 天天操天天干天天综合网 | 午夜av在线播放 | 91麻豆操 | 色欧美88888久久久久久影院 | 亚洲在线高清 | 久久视频一区二区 | 在线免费观看欧美日韩 | 国外成人在线视频网站 | 中文字幕av一区二区三区四区 | 亚洲免费不卡 | 91av在线不卡| 精品在线视频观看 | 中文在线免费看视频 | 久久久久久久久久久久电影 | 亚洲精品在线观 | 亚洲一级黄色av | 亚洲免费av电影 | 91av网址| 在线免费黄色毛片 | 狠狠操夜夜操 | 91高清不卡 | 麻花豆传媒mv在线观看 | 亚洲五月综合 | 在线观看国产日韩欧美 | 成人羞羞视频在线观看免费 | 久久久国产在线视频 | 国精产品满18岁在线 | 久久久久久久久久久综合 | 国产精品观看在线亚洲人成网 | 精品1区2区3区 | 亚洲午夜久久久久久久久电影网 | 精品国产福利在线 |