🌲 😄 主要是建立一个最基础的单元测试示例,包含的测试用例主要包括以下的
🐍 我们的原则还是业务都写在Service里边哦

测试的内容

  1. 测试的内容需要覆盖70%的代码才对,不然就容易出现意想不到的bug
  • Service的测试
  • 包含数据库
  • 包含web请求
  • 包含RPC请求
  • dao的测试
  • Dao的测试其实更多的是为了验证SQL的正确性。也即是为了验证ibatis/mybatis xml语句的正确性,如果可以伪造验证xml生成的sql语句是正确的,那么数据库这一层面也是可以避免掉的。
  • 内部方法的测试
  • 不包含数据库以及其他网络请求的测试

那些是不需要测试的

  • 静态方法是不测试的
  • 那么就需要保证这个静态方法的绝对正确性,输入输出,边界等等
  • 不允许Service的实现出现静态方法,因为不好测试,都放在util里边才是对的

测试的边界

不能按照写代码的思考流程进行测试,外部输入是无穷无尽的,人的行为和动作是无限的.

  • 边界测试
  • 正常测试
  • 异常数据测试

如何考虑数据污染的问题

如何保证我们的数据库数据不被本地测试污染

  • 事务
  • 单数据源
  • 多数据源

Example

  • Mock 标记一个Mock字段
  • Spy
    • Mockito does not delegate calls to the passed real instance, instead it actually creates a copy of it. So if you keep the real instance and interact with it, don’t expect the spied to be aware of those interaction and their effect on real instance state. The corollary is that when an unstubbed method is called on the spy but not on the real instance, you won’t see any effects on the real instance.
  • InjectMocks 为对象注入Mock字段
  • MockitoAnnotations.initMocks(this); 初始化测试用例类中由Mockito的注解标注的所有模拟对象
  • when(chatRecordDubboService.saveChatRecord(Matchers.isA(ChatRecord.class), Matchers.eq(EnumSourceType.KN_CLIENT))).thenReturn(“123123”);

    • 为mock对象和spy对象设置返回值
    • when(service.callMethod(param1,param…)).thenReturn()
    • doNotion().when(service.callMethod(params…))

      1. public class ChatRecordServiceImplTest {
      2. @Mock
      3. StringRedisTemplate redisTemplate;
      4. @Mock
      5. private ValueOperations<String, String> valueOperations;
      6. @Mock
      7. private IChatRecordDubboService chatRecordDubboService;
      8. @Mock
      9. private MongoTemplate mongoTemplate;
      10. @Spy
      11. @InjectMocks
      12. private ChatRecordServiceImpl spy;
      13. @Before
      14. public void setUp() throws Exception {
      15. // 初始化测试用例类中由Mockito的注解标注的所有模拟对象
      16. MockitoAnnotations.initMocks(this);
      17. when(chatRecordDubboService.saveChatRecord(Matchers.isA(ChatRecord.class),
      18. Matchers.eq(EnumSourceType.KN_CLIENT))).thenReturn("123123");
      19. when(mongoTemplate.updateFirst(Matchers.isA(Query.class), Matchers.isA(Update.class), Matchers.eq(MongoTableConst.HISTORY_RECORD)))
      20. .thenReturn(new WriteResult(1, true, "1111"));
      21. when(redisTemplate.opsForValue()).thenReturn(valueOperations);
      22. doNothing().when(valueOperations).set(anyString(), anyString());
      23. doNothing().when(valueOperations).set(anyString(), anyString(), anyInt(), anyObject());
      24. }
      25. @Test
      26. public void testMockDubboService() {
      27. final String s = chatRecordDubboService.saveChatRecord(new ChatRecord(), EnumSourceType.KN_CLIENT);
      28. Assert.assertEquals("123123", s);
      29. }
      30. @Test
      31. public void saveChatRecord() {
      32. KnClientResponse result = new KnClientResponse();
      33. fill(result);
      34. BotResponse response = new BotResponse();
      35. fill(response);
      36. DialogueContext dialogueContext = new DialogueContext();
      37. final String s = spy.saveChatRecord(result, dialogueContext, EnumSpeakerType.BOT, 1, response, "1212");
      38. Assert.assertEquals("123123", s);
      39. }
      40. protected void fill(KnClientResponse knClientResponse) {
      41. knClientResponse.setOpenId(Md5.MD5("随风而走的我"));
      42. knClientResponse.setSourceId("123123");
      43. knClientResponse.setAppId("12312312312312");
      44. List<AnswerDto> answerList = new ArrayList<>();
      45. {
      46. AnswerDto first = new AnswerDto();
      47. first.setTurnSourceType("taskbot");
      48. first.setImageList(null);
      49. first.setTurnItemId("I12313123123");
      50. first.setQuestion("早上好");
      51. first.setAnswer("亲,早上好,请问有什么可以帮助你的吗?");
      52. first.setStatus(0);
      53. first.setScore(200D);
      54. first.setIntentId("I12313123");
      55. first.setRuleName(null);
      56. first.setQaName(null);
      57. first.setPredictSrc("全路径匹配");
      58. first.setMatchKeyWord(null);
      59. answerList.add(first);
      60. }
      61. {
      62. AnswerDto second = new AnswerDto();
      63. second.setTurnSourceType("taskbot");
      64. second.setImageList(null);
      65. second.setTurnItemId("I12313123123");
      66. second.setQuestion("早上好");
      67. second.setAnswer("你好");
      68. second.setStatus(0);
      69. second.setScore(200D);
      70. second.setIntentId("I12313123");
      71. second.setRuleName(null);
      72. second.setQaName(null);
      73. second.setPredictSrc("全路径匹配");
      74. second.setMatchKeyWord(null);
      75. answerList.add(second);
      76. }
      77. knClientResponse.setAnswerList(answerList);
      78. knClientResponse.setQuestion("早上好");
      79. knClientResponse.setAnswerStatus(0);
      80. knClientResponse.setNoticeType(0);
      81. knClientResponse.setMessageId("11111111111");
      82. knClientResponse.setNoticeSource(null);
      83. }
      84. protected void fill(BotResponse botResponse) {
      85. botResponse.setDialogState(null);
      86. botResponse.setResultLevel(EnumResultLevel.GOOD);
      87. List<Map<String, String>> stateVars = new ArrayList<>();
      88. Map<String, String> temp = new HashMap<>();
      89. temp.put("first", "sdsdf");
      90. stateVars.add(temp);
      91. botResponse.setStateVars(stateVars);
      92. botResponse.setRepository(....);
      93. }
      94. }

这样的测试也不依赖数据库,Dubbo服务等等数据,随时都可以启动,随时都可以进行验证的。

体验

单元测试写起来真的很麻烦,需要伪造各种各样的数据,对正确的流程,错误的流程以及边界进行测试。

写完单元测试之后,还要不断的维护测试代码。在某一次重构之后,对参数进行了大的调整,也要对测试代码进行调整。

写测试代码消耗的时间往往比写代码的时间还要长。但是带来的好处也是显而易见的。就是简单的Bug在测试阶段就会被fix掉,剩下的Bug往往是对业务的理解不到位,以及数据库数据错误导致的(这里发生在提测阶段)

参考链接

https://segmentfault.com/a/1190000006746409
https://github.com/hehonghui/mockito-doc-zh