测试范围

西魏陶渊明 ... 2022-3-24 大约 3 分钟

相关信息

项目中拿些类需要进行单元测试呢? 单测不是目的,是手段。目的都是保证质量

# 一、从项目分层维度设计单测用例

常见的项目分层有一下这些。

  • Web层
  • Service层
  • domain层
  • Integration层
  • Common层 公用类必须要做

# 1.1 Web层 【忽略】

一般web层已经上线不会发生改动,往往改动的是新增或者删除参数。可以忽略。

# 1.2 Service层 【建议】

Service层往往是程序的接口层,接受来自Web或者Message、Job的调用。Service的逻辑会覆盖大部分的下游分层。Service是我们的业务入口 接口往往不会改变,改变最多的是它的实现。这一层我们要做好单元测试。

但是这一层的单元测试可能会涉及数据的读写和外部的数据读写。可以使用Mockito数据进行替换外部数据的返回。 内部数据的读写可以通过事务回滚的方式处理,主要验证自己代码中对这些数据的结算逻辑是否有问题。

这一层入参的数据往往会很多,对系统也有相关的依赖,自行评估方案。 这里提供两个曲线救国的方案:

  1. 如果不好测试可以对程序的最小单元进行测试,曲线救国。
  2. 对某个方法中,不满足测试场景的接口,进行Mockito,以跳过这段逻辑。

# 1.3 Domain层 【建议】

这一层和Service的业务是比较相像的,只不过不会有Service层一样那么复杂,参考上文。

# 1.4 Integration层 【建议】

防腐层,主要逻辑就是对外部接口的调用,主要测试不要出现NPE。参数的输入值做好非空非null判断,对返回结果做好非空和非null。

public List<GoodsBaseMsgDTO> querySkuList(Long skuId, Long brandId, String goodsName) {
        if (Objects.isNull(skuId) && Objects.isNull(brandId) && StringUtils.isBlank(goodsName)) {
            // 根据业务来判断是报错还是为空
            return EnhanceStream.emptyList();
        }
        GoodsMsgQueryParam queryParam = new GoodsMsgQueryParam();
        queryParam.setBrandId(brandId);
        queryParam.setSkuId(skuId);
        queryParam.setGoodsName(goodsName);
        JsonResult<PageData<GoodsBaseMsgDTO>> pageDataJsonResult = goodsStockApi.pageQuerySkuList(queryParam);
        PageData<GoodsBaseMsgDTO> pageData = JsonResultUtils.orElseGetSafeData(pageDataJsonResult, new PageData<GoodsBaseMsgDTO>());
        if (Objects.isNull(pageData)) {
            return EnhanceStream.emptyList();
        }
        return pageData.getRecords();
    }
    
public static <T> T orElseGetSafeData(JsonResult<T> result, T defaultValue) {
        if (null != result && result.isSuccess() && Objects.nonNull(result.getData())) {
            return result.getData();
        } else {
            if (Objects.nonNull(result)){
                log.error(result.getMsg());
            }
            return Objects.nonNull(defaultValue) ? defaultValue : null;
        }
}    
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27

外部提供的 Feign 接口, 在没有提供实现之前可以先使用 Mockito进行Mock 帮助完成接口的测试。

public class TradeShopIntegrationImplTest extends BaseApplicationTest {

    @Autowired
    private TradeShopIntegration shopBrandIntegration;

    @MockBean
    private BrandServiceApi brandService;
    
    @MockBean
    private GoodsStockApi goodsStockApi;
    
    @Test
    public void testGetAllBrands() {
        Mockito.doReturn(JsonResult.failure("fail")).when(goodsStockApi).getSkuList(Mockito.any());
        // 底层调用的是goodsStockApi.getSkuList()
        List<GoodsBaseMsgDTO> goodsBaseMsgDTOS = shopBrandIntegration.queryAllSku();
        // 因为前面声明了返回fail。所以这里没有数据返回。
        JsonConsoleUtils.println(goodsBaseMsgDTOS);
        // 这里跟上面的区别就是,如果没有声明返回值,就走原来的方法。
        List<OutBrandDTO> allBrands = shopBrandIntegration.getAllBrands();
        JsonConsoleUtils.println(allBrands);
    }
    
}    

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25

# 1.5 Common层 【必须 】

这一层基本出现的都是工具类,主要是功能性代码,自己提供的工具类,一定要做好单元测试。主要是考虑 异常情况和极限的情况。工具类因为变动的不会太频繁所以维护成本是也比较低的。

Mockito学习跳转 (opens new window)

# 二、针对核心计算逻辑设计单测用例

比较核心的计算逻辑进行测试。比如根据外部数据进行组装生成新的数据模型的方法。

# 三、忽略测试

在实际开发中可能有些测试只是为了验证某些数据,每次执行要提前准备数据之类的。这种特殊的单测,只能人工指定运行的 我们可以忽略它,使用@Ignore 忽略。这样在Maven运行单测时候,会只运行你没有进行忽略的单测。


本文由西魏陶渊明版权所有。如若转载,请注明出处:西魏陶渊明
上次编辑于: 2022年6月16日 21:10
贡献者: lxchinesszz