Loading

单元测试Tips

多个测试用例应该怎么组织?#

假设需要测试的类是A,B,C,D,A和B在pk1包下,B和C在pk2包下
如果每次都手动点击A,B,C,D这几个类对应的测试类,那么很不方便,
我们可以对pk1包下的A和B的测试类分为以下几个,ATest,BTest,Pk1AllTest,其中ATest和BTest分别对应A和B的测试类,Pk1AllTest将ATest和BTest组合进来,执行Pk1AllTest就相当于一起执行了ATest和BTest,具体代码如下

package org.example.pk1;

import org.junit.After;
import org.junit.Assert;
import org.junit.Before;
import org.junit.Test;

public class ATest {
    @Test
    public void yesTest() {
        A a = new A();
        Assert.assertTrue(a.yes());
    }

    @Test
    public void noTest() {
        A a = new A();
        Assert.assertFalse(a.no());
    }
}
public class BTest {

    @Test
    public void yesTest() {
        B a = new B();
        Assert.assertTrue(a.yes());
    }

    @Test
    public void noTest() {
        B a = new B();
        Assert.assertFalse(a.no());
    }
}
@RunWith(Suite.class)
@Suite.SuiteClasses({
        ATest.class,
        BTest.class
})
public class Pk1AllTest {
}

CTest, DTest, Pk2AllTest同理,然后,在pk1和pk2包的上一级,建一个AllTest,AllTest中,把Pk1AllTest和Pk2AllTest注入进来,执行AllTest,就把所有要测试的类都执行了。

@RunWith(Suite.class)
@Suite.SuiteClasses({
        Pk1AllTest.class,
        Pk2AllTest.class
})
public class AllTest {

}

一个重构的例子#

待重构的代码

public class TestRefactor {
    public void execute() {
        // 假设T方法需要很多资源和初始化的动作,不方便构造进行测试
        T t = new T();
        t.begin();
        System.out.println("do your biz");
        t.end();
    }
}

class T {
    public void begin() {
        System.out.println("T begin");
    }

    public void end() {
        System.out.println("T end");
    }
}

假设T方法需要很多资源和初始化的动作,不方便构造进行测试,我们需要Mock T这个类的begin和end方法,此时,我们可以先重构一下代码,重构后的代码如下:

public class TestRefactor {
    public void execute() {
        T t = getT();
        t.begin();
        System.out.println("do your biz");
        t.end();
    }

    protected T getT() {
        return new T();
    }
}

class T {
    public void begin() {
        System.out.println("T begin");
    }

    public void end() {
        System.out.println("T end");
    }
}

那么我们进行单元测试的时候,在新建被测试类的时候,可以传入一个匿名内部类,并重写getT()方法,让其返回我们自定义的MockT对象,这样就实现了用MockT模拟T行为的测试需求。

public class TestRefactorTest {

    @Test
    public void execute() {
        TestRefactor refactor = new TestRefactor() {
            protected T getT() {
                return new MockT();
            }
        };
        refactor.execute();
    }

    public static class MockT extends T {
        @Override
        public void begin() {
            System.out.println("mock method begin");
        }

        @Override
        public void end() {
            System.out.println("mock method end");
        }
    }
}

Mock Object示例#

对于一些方法中使用到了接口方法,比如HttpServletRequest,这些类的实现可能是由具体的容器,比如Tomcat提供的,我们要运行其单元测试就必须部署到Tomcat容器中,非常不方便,这里我们使用Easy Mock,来模拟接口的具体实现方法的返回行为,这样就便于我们进行单元测试了。
引入依赖

   <dependency>
      <groupId>org.easymock</groupId>
      <artifactId>easymock</artifactId>
      <version>4.2</version>
      <scope>test</scope>
    </dependency>

假设需要测试的方法如下:

public class Service {
    public String execute(MyInterface myInterface) {
        String username = myInterface.getParam("username");
        return username;
    }
}

其中MyInterface是接口,需要依赖具体的实现,如果我们模拟一个实现类,那么就需要重写很多方法,现在我们采用mock的方式,仅需要使用这个接口的getParam方法即可,单元测试的代码如下:

public class ServiceTest {

    @Test
    public void execute() {
        MyInterface myInterface = EasyMock.createMock(MyInterface.class);
        EasyMock.expect(myInterface.getParam("username")).andReturn("userName");
        EasyMock.replay(myInterface);
        assertEquals("userName", new Service().execute(myInterface));
    }
}

源码#

Github

Gitee

作者:GreyZeng

出处:https://www.cnblogs.com/greyzeng/p/13334154.html

版权:本作品采用「署名-非商业性使用-相同方式共享 4.0 国际」许可协议进行许可。

你可以在这里自定义其他内容

posted @   Grey Zeng  阅读(328)  评论(0编辑  收藏  举报
编辑推荐:
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
阅读排行:
· 震惊!C++程序真的从main开始吗?99%的程序员都答错了
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理
· 单元测试从入门到精通
· 上周热点回顾(3.3-3.9)
· Vue3状态管理终极指南:Pinia保姆级教程
点击右上角即可分享
微信分享提示
more_horiz
keyboard_arrow_up light_mode palette
选择主题
menu