在Maven多模块项目中进行跨模块的SpringBoot单元测试

项目结构:

lua 复制代码
demo-root(父模块,pom.xml)
├── **demo-common**          -- 通用模块(核心工具类、常量、基础配置)
├── ...
├── **demo-system**          -- 系统模块(用户/角色/权限等核心业务)
└── **demo-app**             -- 启动模块(主应用,依赖其他模块)

demo-app模块下的RootApplication.java:

java 复制代码
package org.example.app;

import org.mybatis.spring.annotation.MapperScan;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;

@SpringBootApplication
@MapperScan("org.example.*.mapper")
@ComponentScan("org.example")
public class RootApplication {

    public static void main(String[] args) {
        SpringApplication.run(RootApplication.class, args);
    }

}

demo-system模块下的mapper层单元测试类:

java 复制代码
package org.example.system.mapper;

import org.example.system.domain.entity.User;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;

import java.util.List;

import static org.junit.jupiter.api.Assertions.*;

@SpringBootTest(classes = org.example.app.RootApplication.class) // 跨模块需指定启动类
class UserMapperTest {

    @Autowired
    private UserMapper userMapper;

    @Test
    void test() {
        List<User> users = userMapper.selectList(null);
        System.out.println(users);
    }
}

demo-system模块的pom.xml:

xml 复制代码
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <parent>
        <artifactId>demo-root</artifactId>
        <groupId>org.example</groupId>
        <version>0.0.1-SNAPSHOT</version>
    </parent>
    <modelVersion>4.0.0</modelVersion>

    <artifactId>demo-system</artifactId>

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

    <dependencies>
        <dependency>
            <groupId>org.example</groupId>
            <artifactId>demo-app</artifactId>
            <version>0.0.1-SNAPSHOT</version>
        </dependency>
        <dependency>
            <groupId>org.example</groupId>
            <artifactId>demo-common</artifactId>
            <version>0.0.1-SNAPSHOT</version>
        </dependency>
    </dependencies>
</project>

因为User实体继承自BaseEntity,所以为了获取到完整的User,打印继承字段和非继承字段,所以需要增加lombok配置。

在实体类中添加几个注解。

java 复制代码
package org.example.system.domain.entity;

import com.baomidou.mybatisplus.annotation.TableName;
import lombok.Data;
import lombok.EqualsAndHashCode;
import lombok.ToString;
import org.example.common.model.BaseEntity;

@TableName("users")
@Data
@EqualsAndHashCode  // lombok注解,自动处理继承字段。已在lombok.config配置callSuper=True
@ToString   // lombok注解,自动处理继承字段。已在lombok.config配置callSuper=True
public class User extends BaseEntity {
    private String username;
}

在根目录下新建一个lombok.config文件:

bash 复制代码
# 自动为所有 @ToString 添加 callSuper=true
lombok.toString.callSuper=CALL

# 自动为所有 @EqualsAndHashCode 添加 callSuper=true
lombok.equalsAndHashCode.callSuper=CALL

demo-system模块下的service层单元测试类:

java 复制代码
package org.example.system.service;

import org.example.system.domain.entity.User;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;

import java.util.List;

import static org.junit.jupiter.api.Assertions.*;

@SpringBootTest(classes = org.example.app.RootApplication.class)
public class IUserServiceTest {

    @Autowired
    private IUserService userService;

    @Test
    void test() {
        List<User> list = userService.list();
        System.out.println(list);
    }
}

遇到的其他问题:在运行demo-system模块下的service层单元测试类时,demo-app模块下的启动类扫到了demo-common模块下的JWT工具类显示配置文件读取失败,使用@Value注解的字段无法读取正确的参数值。

bash 复制代码
Caused by: java.lang.IllegalArgumentException: Could not resolve placeholder 'jwt.secret' in value "${jwt.secret}"

原因分析:所有模块的配置文件都叫application.yml,在正式运行时,只会加载读取启动模块的配置文件,并且, 单元测试不会自动读取src/main/resources下的配置文件,而是优先从src/test/resources/加载。

解决方案:对多模块项目的配置文件,非启动模块的配置文件采用application-xxx.yml的命名方式,并在启动模块下的配置文件中激活,确保所有模块的配置文件都可以被正常加载读取。

application-common.yml:

yml 复制代码
jwt:
  secret: "" # 建议用Base64编码
  # JWT 的签名算法(如 HS256)要求密钥是 256 位(32 字节)。Base64 编码可以确保密钥长度正确,避免因密钥长度不足导致的安全问题。
  expiration: 86400 # Token有效期(秒),默认1天(JwtUtils 中的 expiration 单位是秒
spring:
  profiles: common

application.yml:

yml 复制代码
server:
  port: 8080
spring:
  profiles:
    include: common, system

为使单元测试正常,直接复制主配置文件到测试资源目录:src/test/resources/application.yml


总结:

在Maven多模块项目中进行跨模块的SpringBoot单元测试的几个要点。

第一,在启动类中指定组件扫描路径和mapper包路径,引入启动模块依赖,在单元测试类中指定启动类。

第二,使用实体基类需添加lombok配置。

第三,差异化命名各模块的配置文件,并在启动模块的配置文件中激活。最后复制主配置文件到测试资源目录。


补充:后为正常运行启动类,我把写了数据库连接配置的application-system.yml放到启动模块下。

相关推荐
创码小奇客25 分钟前
从 0 到 1 落地 SpringBoot+RocketMQ:架构师亲授分布式通信最优解
spring boot·rocketmq·trae
小醉你真好39 分钟前
Spring Boot + ShardingSphere 实现分库分表 + 读写分离实战
spring boot·后端·mysql
斜月1 小时前
Spring 自动装配原理即IOC创建流程
spring boot·后端·spring
半部论语2 小时前
Spring **${}** vs **#{}** 语法全景图
java·数据库·spring boot·后端·spring
麦兜*3 小时前
Spring Integration 整合 Web3.0网关:智能合约事件监听与Spring Integration方案
java·spring boot·后端·spring·spring cloud·web3·智能合约
日月星辰Ace3 小时前
🚀 从压测到优化:用 k6 和 SpringBoot 构建高性能系统(附实战案例)
spring boot·后端·性能优化
天机️灵韵5 小时前
开源医院信息管理系统:基于若依框架的智慧医疗解决方案
java·开发语言·spring boot·spring cloud·github·开源项目
Resean02236 小时前
SpringMVC 6+源码分析(三)DispatcherServlet实例化流程 2--(url 与contrller类如何进行映射)
java·spring boot·spring
2025年一定要上岸6 小时前
【Django】-10- 单元测试和集成测试(下)
数据库·后端·python·单元测试·django·集成测试
MrSYJ7 小时前
WebSecurityConfigurerAdapter自动调用configure(HttpSecurity http)
spring boot