不同系统下打包SpringBoot项目,单元测试类的执行顺序不同导致的问题

项目背景:

正式环境项目上线前跑单元测试发现一直报 NullPointerException

项目结构如下:

配置文件如下:

    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.0.6.RELEASE</version>
        <relativePath/>
    </parent>

    <properties>
        <java.version>11</java.version>
        <maven.compiler.source>11</maven.compiler.source>
        <maven.compiler.target>11</maven.compiler.target>
    </properties>

    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <scope>test</scope>
        </dependency>
    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-surefire-plugin</artifactId>
            </plugin>
        </plugins>
    </build>

完整代码如下:

@RestController
@RequestMapping("/example")
public class ExampleController {

    @Autowired
    private ExampleManager exampleManager;

    public void test() {
        exampleManager.test();
    }

    public void setExampleManager(ExampleManager exampleManager) {
        this.exampleManager = exampleManager;
    }
}
@Component
public class ExampleManager {

    @Autowired
    private ImportManager importManager;

    public void test() {
        importManager.test();
    }
}
@Component
public class ImportManager {

    public void test() {
    }
}
public class MockExampleManager extends ExampleManager {
}
@RunWith(SpringRunner.class)
@SpringBootTest(classes = StartServer.class)
public class GoodTest {

    private Logger log = LoggerFactory.getLogger(GoodTest.class);

    @Autowired
    private ExampleController exampleController;
    MockExampleManager mockExampleManager;

    @Before
    public void before() {
        log.info("执行{}.before", this.getClass().getName());
        mockExampleManager = new MockExampleManager();
        exampleController.setExampleManager(mockExampleManager);
    }

    @Test
    public void test() {
        exampleController.test();
    }


@RunWith(SpringRunner.class)
@SpringBootTest(classes = StartServer.class)
public class OrderTest {

    private Logger log = LoggerFactory.getLogger(OrderTest.class);

    @Autowired
    private ExampleController exampleController;
    MockExampleManager mockExampleManager;

    @Before
    public void before() {
        log.info("执行{}.before", this.getClass().getName());
        mockExampleManager = new MockExampleManager();
        exampleController.setExampleManager(mockExampleManager);
    }

    @Test
    public void test() {
        exampleController.test();
    }
}

问题描述:

SpringBoot项目本地打包未报错,测试环境Jenkins打包即报错。

提取关键代码如下:

@Component
public class ExampleManager {

    @Autowired
    private ImportManager importManager;
    
    public void test() {
        importManager.test();
    }
}

以上方法的 importManager.test() 在Jenkins上打包时总是会报 NullPointerException。在本地打包时就不会。


原因分析:

第一步:找到 Jenkins 上报错的类和方法 ExampleManager.test(),通过本地 Debug 发现单元测试运行到 ExampleManager.test() 方法时,属性 importManager 并不是 Spring 容器中的 ImportManager 类实例,而是 MockExampleManager 类实例。

第二步:查找用到 MockExampleManager 类的地方,发现 OrderTest 类用到了,于是注释了 OrderTestMockExampleManager 类的代码。此时本地打包不再报错,但是 Jenkins 打包依然报错。

猜测到是其他类注入了 ExampleController 并修改了依赖的 ExampleManager 实例。

第三步:注释 GoodTestMockExampleManager 类的代码。此时本地和 Jenkins 打包都不再报错。

查询资料后分析原因应该是不同操作系统下文件排序会有差异,导致两个单元测试类的执行顺序有差异。
又由于两个单元测试类都注入了 ExampleController 实例,这个实例是单例的,一个地方对 ExampleController 实例属性的修改会影响其他地方。

在我本地因为是 OrderTest 先执行,所以 GoodTest 中的 exampleController.setExampleManager(exampleManager) 代码影响不到 OrderTest

如下图所示,我本地的运行效果:

在这里插入图片描述


解决方案:

找到用 MockExampleManager 类的地方,发现 GoodTestOrderTest 类中的 exampleController.setExampleManager(mockExampleManager) 代码是没有必要的,因此删除,就解决了 NullPointerException 异常。

没有 NullPointerException 异常,在不同的平台打包都不会出错啦。

  • 10
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring Boot项目中,可以通过创建一个新的单元测试来进行单元测试。在这个中,你需要添加`@SpringBootTest`注解来表示当前单元测试运行在Spring Boot环境中。同时,你还需要导入相关的依赖和,例如`org.junit.jupiter.api.Assertions`和`org.junit.jupiter.api.Test`。在这个单元测试中,你可以编写测试方法来测试你的业务代码。例如,你可以使用`@Test`注解来标记一个测试方法,并在方法中编写测试逻辑。在测试方法中,你可以使用断言来验证你的代码的预期行为。例如,你可以使用`Assertions.assertNotNull()`来验证某个对象不为空。以下是一个示例的单元测试: ```java import com.example.demo.mapper.UserMapper; import com.example.demo.model.UserInfo; import org.junit.jupiter.api.Assertions; import org.junit.jupiter.api.Test; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.test.context.SpringBootTest; @SpringBootTest class UserMapperTest { @Autowired private UserMapper userMapper; @Test void getUserById() { UserInfo userInfo = userMapper.getUserById(1); Assertions.assertNotNull(userInfo); } } ``` 在这个示例中,我们使用`@Autowired`注解来自动注入`UserMapper`对象,并在`getUserById()`方法中调用`userMapper.getUserById()`方法来获取用户信息。然后,我们使用`Assertions.assertNotNull()`来验证获取到的用户信息不为空。这样,我们就可以通过运行这个单元测试来测试`UserMapper`的功能。 #### 引用[.reference_title] - *1* *2* *3* [【SpringBootSpringBoot单元测试](https://blog.csdn.net/m0_46468731/article/details/125969337)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值