• 代码整洁之道-读书笔记之单元测试


    TDD:测试驱动开发,先写测试,再写逻辑代码,通过单测,写逻辑代码,依次循环,知道所有逻辑都完成

    1. TDD 三定律

    第一定律:在编写不能通过的单元测试前,不可变写生产代码。

    第二定律:只可编写刚好不可通过的单元测试,不能编译也算不通过。

    第三定律:只可编写刚好足以通过当前失败测试的生产代码。

    2. 保持测试整洁

    有人认为测试代码不用遵循生产代码的质量标准

    1. 变量名不用好
    2. 测试函数不必短小和具有描述
    3. 测试代码不必做良好的设计和仔细划分
    4. 测试代码只要还能工作即可

    不好的测试带来的问题

    1. 后期修改成本高
    2. 新增测试代码难度大
    3. 可能需要丢弃测试代码,直接使用无测试的代码,风险高

    所以测试代码和生产代码一样重要。

    测试的好处

    保证生产代码的可扩展、可维护、可复用

    改着放心,用着舒心

    3. 整洁的测试

    可读性是测试代码最重要的衡量指标,要保证测试代码的明确、简介、还有足够的表达力

    测试的三个环节

    give(准备数据)

    when ( 执行逻辑)

    then (验证逻辑)

    看一个例子

    package com.xxx.mas.cd.platform.service;
    
    import com.xxx.mas.cd.platform.TestHelper;
    import com.xxx.mas.cd.platform.common.config.AppPipelineConfiguration;
    import com.xxx.mas.cd.platform.controller.dto.pipeline.AndroidAppPipelineGenerateFrontParamRequest;
    import com.xxx.mas.cd.platform.controller.dto.pipeline.AppPipelineGenerateFrontParamResponse;
    import com.xxx.mas.cd.platform.service.android.AndroidAppBuildParamConfigService;
    import com.xxx.mas.cd.platform.service.android.AndroidAppBuildService;
    import com.xxx.mas.cd.platform.service.android.AndroidBaseProjectBuildConfigParamService;
    import com.xxx.mas.cd.platform.service.android.AndroidProjectRegisterService;
    import com.xxx.mas.cd.platform.service.pipeline.AndroidAppPipelineConfigService;
    import com.xxx.mas.common.entity.dto.pipeline.AppPipelineConfigGeneratorDTO;
    import com.xxx.mas.common.proxy.GitProxy;
    import com.xxx.mas.common.utils.JsonUtils;
    import org.apache.velocity.app.Velocity;
    import org.junit.Before;
    import org.junit.Test;
    import org.junit.runner.RunWith;
    import org.mockito.InjectMocks;
    import org.mockito.Mock;
    import org.mockito.MockitoAnnotations;
    import org.powermock.api.mockito.PowerMockito;
    import org.powermock.modules.junit4.PowerMockRunner;
    
    import java.io.IOException;
    
    import static org.assertj.core.api.Assertions.assertThat;
    import static org.mockito.ArgumentMatchers.any;
    import static org.powermock.api.mockito.PowerMockito.doReturn;
    import static org.powermock.api.mockito.PowerMockito.when;
    
    @RunWith(PowerMockRunner.class)
    public class AndroidAppPipelineConfigServiceTest {
    	@Mock
    	private AppPipelineConfiguration appPipelineConfiguration;
    	@Mock
    	private GitProxy gitProxy;
    	@Mock
    	private AndroidProjectRegisterService androidProjectRegisterService;
    	@Mock
    	private AndroidBaseProjectBuildConfigParamService androidBaseProjectBuildConfigParamService;
    	@Mock
    	private AndroidAppBuildParamConfigService androidAppBuildParamConfigService;
    	@Mock
    	private AndroidAppBuildService androidAppBuildService;
    	@InjectMocks
    	private AndroidAppPipelineConfigService androidAppPipelineConfigService;
    
    	@Before
    	public void init() {
    		MockitoAnnotations.initMocks(this);
    		Velocity.init();
    	}
    
    	@Test
    	public void testGenerateAppPipelineFrontTemplate() throws IOException {
    		AndroidAppPipelineGenerateFrontParamRequest param = AndroidAppPipelineGenerateFrontParamRequest.builder()
    				.applicationId("com.xxx.calculator")
    				.appPipelineBuildType("test")
    				.build();
    		String templateParam = "{\"git\":{\"url\":\"git@git.xxxx.com:QaTools/calculator.git\",\"branch\":\"master\"},\"type\":\"test\",\"application_id\":\"com.xxx.calculator\",\"gradlew_path\":\"gradlew\",\"project_build_file_root_path\":\"build.gradle\",\"module_build_file_path\":\"app/build.gradle\",\"app_module\":\"app\",\"build_variant\":\"fullRelease\",\"gradle_cache_dir\":null}";
    		String androidPipeline = TestHelper.getResource("android_pipeline_param.json");
    		String expectResult = TestHelper.getResource("android_generate_pipeline_front_template.json");
    		AppPipelineConfigGeneratorDTO appPipelineConfigGenerator = AppPipelineConfigGeneratorDTO.builder().templateParam(templateParam).build();
    
    		AndroidAppPipelineConfigService androidAppPipelineConfigServiceSpy = PowerMockito.spy(androidAppPipelineConfigService);
    		doReturn(appPipelineConfigGenerator).when(androidAppPipelineConfigServiceSpy).processAppPipelineConfig(param);
    		when(gitProxy.getFileContent(any(), any(), any())).thenReturn(androidPipeline);
    
    		AppPipelineGenerateFrontParamResponse result = androidAppPipelineConfigServiceSpy.generateAppPipelineFrontTemplate(param);
    
    		assertThat(JsonUtils.objectToJson(result)).isEqualTo(expectResult);
    	}
    
    }
    
    • 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
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75

    4.每一个测试必须有断言

    也就是我上面说的then这个阶段,如果我们只是进行执行逻辑,缺不严重逻辑,这样的测试也就没什么意义了

    每一个测试只测一种case

    开发中一个方法可能存在多种业务逻辑,我们应该每一种case进行一次单独测试编写

    5.R.I.R.S.T

    测试遵循的5条规则

    快速:测试代码应该执行快,以便于频繁的执行测试

    独立:测试和测试之间应该相互独立,不能一个测试作为另一个测试的条件,每一个测试都是可以独立运行的

    可重复:测试应该是可以重复执行的,而且获的结果是一致的

    自足验证: 测试一定是有断言的,可以自我验证逻辑的正确性

    及时:测试应该和生产代码同时编写。

  • 相关阅读:
    python连接sqlserver
    代码随想录Day21 回溯 LeetCodeT216 组合总和III LeetCode T17电话号码的字母总和
    MIT6.824 2022 Raft
    【javaEE】网络原理(网络层)
    热修复技术可谓是百花齐放
    MySQL系统变量之lc_time_names语言环境
    照着这个保姆级文档来,虚拟机装CentOS不再踩坑
    OpenART mini使用教程
    【HMS core】【ML kit】机器学习服务常见问题FAQ
    MATLAB | 艺术就是画圈圈
  • 原文地址:https://blog.csdn.net/constant_rain/article/details/127888745