MyBatis-Plus 单元测试及 H2 数据库应用

MyBatis-Plus 单元测试及 H2 数据库应用

在进行 DAO 层单元测试时,如果使用 @SpringBootTest 启动整个项目进行集成测试,会导致启动速度非常慢,影响测试效率。本文整理了 MyBatis-Plus 单元测试的方法,并介绍如何使用 H2 数据库进行测试,以提高测试效率。

使用 H2 数据库进行单元测试

H2 数据库是一个内存数据库,适用于单元测试场景。可以使用 @MybatisPlusTest 注解来快速配置 MyBatis-Plus 的测试环境。

基本测试配置

less 复制代码
@MybatisPlusTest
@Sql(scripts = {"classpath:db/schema.sql", "classpath:db/data.sql"})
@MapperScan({"mapper包"})
//@ContextConfiguration(classes={*Service.class}) // 如果测试 Service,可以仅加载该 Service 类
public class Test {
    // 单元测试代码
}

注解解析

  • @MapperScan:指定 MyBatis Mapper 接口所在的包,使其在编译后生成对应的实现类。
  • @Sql:测试前执行 SQL 脚本进行数据库初始化。
  • @ContextConfiguration:如果需要加载 ApplicationContext 的配置文件或配置类,可以使用该注解。例如,加载 yml 配置文件时可添加 initializers = {ConfigFileApplicationContextInitializer.class}

使用 Java 配置 H2 数据源

除了使用默认 H2 数据源外,我们也可以手动配置 H2 数据库。

typescript 复制代码
public class DataSourceConfig {
    @Bean
    public DruidDataSource dataSource() {
        DruidDataSource dataSource = new DruidDataSource();
        dataSource.setUrl("jdbc:h2:mem:testdb;DB_CLOSE_DELAY=-1");
        return dataSource;
    }

    @Bean
    public PlatformTransactionManager transactionManager() {
        return new DataSourceTransactionManager(dataSource());
    }
}

该配置类未添加 @Configuration,以避免影响到其他单元测试的数据源。在单元测试类中可以用 @Import 引入该配置类:

less 复制代码
@ExtendWith(SpringExtension.class)
@Import({DataSourceConfig.class})
public class JunitTest {
    @Autowired
    private UserMapper userMapper;

    @Test
    public void testInsertUser() {
        User user = new User();
        user.setName("Test User");
        user.setAge(25);
        userMapper.insert(user);

        User insertedUser = userMapper.selectById(user.getId());
        assertNotNull(insertedUser);
        assertEquals("Test User", insertedUser.getName());
    }
}

如果需要在测试时实时观察 H2 数据库的数据变化,可以使用服务器模式,修改 URL 为:

bash 复制代码
jdbc:h2:tcp://localhost:9092/~/testdb;DB_CLOSE_DELAY=-1

启动 H2 Server 进行调试

如果希望在本地连接 H2 数据库,可以启动 H2 服务器。

csharp 复制代码
public class H2ServerTest {

    private static Server h2Server;

    public static void startH2Server() throws Exception {
        // 启动 H2 服务器
        h2Server = Server.createTcpServer("-tcp", "-tcpAllowOthers", "-tcpPort", "9092").start();
        System.out.println("H2 server started on port 9092");
    }

    public static void stopH2Server() {
        // 停止 H2 服务器
        if (h2Server != null) {
            h2Server.stop();
            System.out.println("H2 server stopped");
        }
    }
}

使用 MySQL 作为测试数据库

如果需要使用应用的 MySQL 数据库进行单元测试,可以添加以下注解:

less 复制代码
@Transactional // 确保测试后回滚数据,避免污染数据库
@AutoConfigureTestDatabase(replace = AutoConfigureTestDatabase.Replace.NONE) // 使用实际数据库,而非默认的 H2
@MybatisPlusTest
@Sql(scripts = {"classpath:db/schema.sql", "classpath:db/data.sql"})
@MapperScan({"mapper包"})
//@ContextConfiguration(classes={*Service.class})
public class Test {
    // 单元测试代码
}

注解解析

  • @Transactional:确保测试后回滚数据库,防止污染数据。
  • @AutoConfigureTestDatabase(replace = AutoConfigureTestDatabase.Replace.NONE):避免 @MybatisPlusTest 默认使用 H2 数据库,而改用 MySQL。

总结

使用 H2 数据库进行 MyBatis-Plus 单元测试可以显著提升测试效率,并避免影响实际数据库。对于更接近生产环境的测试,可以结合 @Transactional@AutoConfigureTestDatabase 直接使用 MySQL 数据库,同时确保数据回滚,避免污染环境。

相关推荐
bing_1581 小时前
如何利用 Redis 的原子操作(INCR, DECR)实现分布式计数器?
数据库·redis·分布式
万能小锦鲤1 小时前
《软件测试与质量控制》实验报告二 单元测试
junit·eclipse·单元测试·测试用例·实验报告·软件测试与质量控制
一个人的博客@你2 小时前
C#多数据库批量执行脚本工具
数据库·c#·批量执行·sql脚本·多数据库执行
果冻kk2 小时前
MySQL MVCC:并发神器背后的原理解析
数据库·mysql
卍郝凝卍2 小时前
云上服务器常见的存储方式和类型
大数据·服务器·数据库
axban3 小时前
QT中删除控件的注意事项、deleteLater和delete的区别
java·数据库·qt
喜欢敲代码的程序员3 小时前
SpringBoot+Mybatis+MySQL+Vue+ElementUI前后端分离版:日志管理(四)集成Spring Security
spring boot·mysql·spring·vue·mybatis
Resean02233 小时前
SpringMVC 6+源码分析(二)DispatcherServlet实例化流程 1
java·spring boot·spring·servlet·springmvc
♪张三儿℡4 小时前
Oracle优化学习十六
数据库·学习·oracle
胡琦博客4 小时前
LLM Prompt与开源模型资源(3)如何写一个好的 Prompt
数据库·开源·prompt