李锋镝的博客

  • 首页
  • 时间轴
  • 评论区显眼包🔥
  • 左邻右舍
  • 博友圈
  • 关于我
    • 关于我
    • 另一个网站
    • 我的导航站
    • 网站地图
    • 赞助
  • 留言
  • 🚇开往
Destiny
自是人生长恨水长东
  1. 首页
  2. 后端
  3. 正文

使用内存数据库进行MyBatis单元测试

2025年12月24日 104点热度 0人点赞 0条评论

使用内存数据库(如H2)进行MyBatis功能测试,能实现轻量、高效、隔离的测试环境(无需部署真实数据库,测试后数据自动销毁)。

一、核心优势

  • 轻量无依赖:无需安装/部署数据库,随用随建;
  • 速度快:内存操作,测试执行效率远高于真实数据库;
  • 数据隔离:每个测试用例可独立初始化数据,无跨用例污染;
  • 无残留:测试结束后内存释放,无需手动清理数据。

二、选型推荐

优先选择 H2数据库:

  • 支持内存模式(mem:)、文件模式;
  • 兼容MySQL/Oracle等主流数据库语法(通过MODE参数);
  • 集成简单,与MyBatis/Spring Boot适配性好。

三、实操步骤(主流场景:Spring Boot + MyBatis)

1. 引入核心依赖(Maven)

<dependencies>
    <!-- MyBatis + Spring Boot整合包 -->
    <dependency>
        <groupId>org.mybatis.spring.boot</groupId>
        <artifactId>mybatis-spring-boot-starter</artifactId>
        <version>3.0.3</version>
    </dependency>
    <!-- H2内存数据库 -->
    <dependency>
        <groupId>com.h2database</groupId>
        <artifactId>h2</artifactId>
        <scope>runtime</scope>
    </dependency>
    <!-- Spring Boot测试包(JUnit 5) -->
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-test</artifactId>
        <scope>test</scope>
    </dependency>
</dependencies>

2. 配置H2数据源与MyBatis

在src/test/resources/application.yml中配置:

spring:
  # H2数据源配置
  datasource:
    # 内存模式:testdb为数据库名;MODE=MySQL兼容MySQL语法;DB_CLOSE_DELAY=-1防止连接关闭后数据库销毁
    url: jdbc:h2:mem:testdb;MODE=MySQL;DB_CLOSE_DELAY=-1;DB_CLOSE_ON_EXIT=FALSE
    driver-class-name: org.h2.Driver
    username: sa  # 默认用户名
    password:     # 默认密码为空
  # 开启H2控制台(可选,方便调试)
  h2:
    console:
      enabled: true  # 访问地址:http://localhost:8080/h2-console
      path: /h2-console
# MyBatis配置
mybatis:
  mapper-locations: classpath:mapper/*.xml  # Mapper XML文件位置
  type-aliases-package: com.example.demo.entity  # 实体类别名包
  configuration:
    log-impl: org.apache.ibatis.logging.stdout.StdOutImpl  # 开启SQL日志(方便调试)

3. 初始化测试表和数据

Spring Boot会自动执行src/test/resources下的schema.sql(建表)和data.sql(插测试数据):

schema.sql(建表脚本):

DROP TABLE IF EXISTS user;
CREATE TABLE user (
    id BIGINT AUTO_INCREMENT PRIMARY KEY,
    name VARCHAR(255) NOT NULL,
    age INT COMMENT '年龄'
);

data.sql(测试数据):

INSERT INTO user (name, age) VALUES ('张三', 20);
INSERT INTO user (name, age) VALUES ('李四', 25);

4. 编写核心代码(实体+Mapper)

(1)实体类 User.java
package com.example.demo.entity;

public class User {
    private Long id;
    private String name;
    private Integer age;

    // 省略getter/setter/toString
}
(2)Mapper接口 UserMapper.java

支持注解/XML两种方式,这里用注解示例:

package com.example.demo.mapper;

import com.example.demo.entity.User;
import org.apache.ibatis.annotations.*;

import java.util.List;

public interface UserMapper {
    @Select("SELECT * FROM user WHERE id = #{id}")
    User getById(Long id);

    @Select("SELECT * FROM user")
    List<User> listAll();

    @Insert("INSERT INTO user (name, age) VALUES (#{name}, #{age})")
    @Options(useGeneratedKeys = true, keyProperty = "id") // 自增主键回填
    int insert(User user);

    @Update("UPDATE user SET name = #{name}, age = #{age} WHERE id = #{id}")
    int update(User user);

    @Delete("DELETE FROM user WHERE id = #{id}")
    int delete(Long id);
}

5. 编写测试用例

核心:用@SpringBootTest启动Spring上下文,@Transactional保证测试后数据回滚(避免污染)。

package com.example.demo.mapper;

import com.example.demo.entity.User;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.transaction.annotation.Transactional;

import java.util.List;

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

@SpringBootTest  // 启动Spring Boot上下文
@Transactional   // 测试方法执行后自动回滚事务,数据不残留
public class UserMapperTest {

    @Autowired
    private UserMapper userMapper;

    // 测试查询单个
    @Test
    void getById() {
        User user = userMapper.getById(1L);
        assertNotNull(user);
        assertEquals("张三", user.getName());
        assertEquals(20, user.getAge());
    }

    // 测试查询全部
    @Test
    void listAll() {
        List<User> users = userMapper.listAll();
        assertEquals(2, users.size());
    }

    // 测试新增
    @Test
    void insert() {
        User user = new User();
        user.setName("王五");
        user.setAge(30);
        int rows = userMapper.insert(user);

        assertEquals(1, rows);          // 验证插入行数
        assertNotNull(user.getId());    // 验证主键回填
        assertEquals(3, user.getId());  // 自增ID:1、2已存在,新ID为3
    }

    // 测试更新
    @Test
    void update() {
        User user = userMapper.getById(1L);
        user.setName("张三更新");
        user.setAge(21);
        int rows = userMapper.update(user);

        assertEquals(1, rows);
        User updatedUser = userMapper.getById(1L);
        assertEquals("张三更新", updatedUser.getName());
    }

    // 测试删除
    @Test
    void delete() {
        int rows = userMapper.delete(1L);
        assertEquals(1, rows);
        assertNull(userMapper.getById(1L));
    }
}

四、纯MyBatis场景(无Spring)

若未使用Spring,需手动管理SqlSession和数据库初始化:

1. 配置mybatis-config.xml

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE configuration
        PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
    <environments default="development">
        <environment id="development">
            <transactionManager type="JDBC"/>
            <dataSource type="POOLED">
                <property name="driver" value="org.h2.Driver"/>
                <property name="url" value="jdbc:h2:mem:testdb;MODE=MySQL;DB_CLOSE_DELAY=-1"/>
                <property name="username" value="sa"/>
                <property name="password" value=""/>
            </dataSource>
        </environment>
    </environments>
    <mappers>
        <mapper resource="mapper/UserMapper.xml"/> <!-- Mapper XML路径 -->
    </mappers>
</configuration>

2. 编写测试用例

package com.example.test;

import com.example.entity.User;
import com.example.mapper.UserMapper;
import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;

import java.io.IOException;
import java.io.InputStream;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.Statement;

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

public class UserMapperTest {

    private SqlSessionFactory sqlSessionFactory;
    private SqlSession sqlSession;
    private UserMapper userMapper;

    // 每个测试方法执行前初始化:加载配置+创建表+插数据
    @BeforeEach
    void setUp() throws IOException, Exception {
        // 1. 加载MyBatis配置
        InputStream inputStream = Resources.getResourceAsStream("mybatis-config.xml");
        sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);

        // 2. 手动初始化H2表和数据
        try (Connection conn = DriverManager.getConnection("jdbc:h2:mem:testdb;MODE=MySQL;DB_CLOSE_DELAY=-1", "sa", "")) {
            Statement stmt = conn.createStatement();
            stmt.execute("DROP TABLE IF EXISTS user;");
            stmt.execute("CREATE TABLE user (id BIGINT AUTO_INCREMENT PRIMARY KEY, name VARCHAR(255), age INT);");
            stmt.execute("INSERT INTO user (name, age) VALUES ('张三', 20), ('李四', 25);");
        }

        // 3. 获取SqlSession(自动提交事务)
        sqlSession = sqlSessionFactory.openSession(true);
        userMapper = sqlSession.getMapper(UserMapper.class);
    }

    // 每个测试方法执行后关闭SqlSession
    @AfterEach
    void tearDown() {
        if (sqlSession != null) {
            sqlSession.close();
        }
    }

    @Test
    void getById() {
        User user = userMapper.getById(1L);
        assertNotNull(user);
        assertEquals("张三", user.getName());
    }
}

五、关键注意事项

  1. 语法兼容:
    H2默认语法与MySQL/Oracle有差异,需通过MODE参数适配:

    • MySQL:jdbc:h2:mem:testdb;MODE=MySQL
    • Oracle:jdbc:h2:mem:testdb;MODE=Oracle
  2. 数据隔离:

    • Spring场景:用@Transactional回滚测试数据;
    • 纯MyBatis场景:每个测试方法重新初始化数据库(@BeforeEach执行建表/插数据)。
  3. H2控制台调试:
    启动应用后访问http://localhost:8080/h2-console,输入配置的JDBC URL(如jdbc:h2:mem:testdb),用户名sa,密码为空,即可查看测试数据。

  4. 日志调试:
    开启MyBatis SQL日志(见application.yml配置),可直观看到执行的SQL和参数,快速定位问题。

六、总结

核心流程:
引入H2依赖 → 配置内存数据源 → 初始化测试表/数据 → 编写Mapper → 编写测试用例(验证CRUD)。
通过内存数据库,能快速搭建隔离的测试环境,确保MyBatis的SQL逻辑、映射关系正确,且测试执行效率极高。

除非注明,否则均为李锋镝的博客原创文章,转载必须以链接形式标明本文链接

本文链接:https://www.lifengdi.com/hou-duan/4666

相关文章

  • org.apache.ibatis.plugin.Interceptor类详细介绍及使用
  • 数据库更新如何实现乐观锁
  • MyBatis vs Spring Data JPA 从原理到实战全解析
  • mybatis-plus-join-boot-starter介绍及用法
  • JDK25模块级导入深度解析:Java导入机制的革命性进化
本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可
标签: MyBatis 内存数据库 单元测试 数据库
最后更新:2025年12月24日

李锋镝

既然选择了远方,便只顾风雨兼程。

打赏 点赞
< 上一篇
下一篇 >

文章评论

1 2 3 4 5 6 7 8 9 11 12 13 14 15 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 46 47 48 49 50 51 52 53 54 55 57 58 60 61 62 63 64 65 66 67 69 72 74 76 77 78 79 80 81 82 85 86 87 90 92 93 94 95 96 97 98 99
取消回复

秋天是倒放的春天,晚安是爱你的序篇。

那年今日(02月10日)

  • 1953年:穆罕默德·纳吉布出任埃及总统
  • 1923年:德国物理学家、X射线发现者伦琴逝世
  • 1898年:德国戏剧家贝尔托·布莱希特出生
  • 1894年:英国政治家哈罗德·麦克米伦出生
  • 589年:杨坚灭陈朝,南北朝结束
  • 更多历史事件
最新 热点 随机
最新 热点 随机
Apollo配置中心中的protalDB的作用是什么 org.apache.ibatis.plugin.Interceptor类详细介绍及使用 JDK25模块级导入深度解析:Java导入机制的革命性进化 AI时代,个人技术博客的出路在哪里? 什么是Meta Server? 千万级大表新增字段实战指南:告别锁表与业务中断
玩博客的人是不是越来越少了?AI时代,个人技术博客的出路在哪里?准备入手个亚太的ECS,友友们有什么建议吗?使用WireGuard在Ubuntu 24.04系统搭建VPNWordPress实现用户评论等级排行榜插件WordPress网站换了个字体,差点儿把样式换崩了
JWT、Cookie、Session、Token 区别与实战选型指南 Spring Boot 2.5.0重新设计的spring.sql.init 配置有啥用? 微服务的数据库设计 MySQL数据库详解——执行SQL更新时,其底层经历了哪些操作? AI重构开发者工作范式:从Anthropic内部调研看Claude对研发领域的深层影响 使用Spring MVC的websocket配置时 Tomcat启动报错
标签聚合
Spring K8s docker JAVA JVM 分布式 数据库 SpringBoot AI IDEA Redis 日常 AI编程 MySQL 多线程 SQL 设计模式 WordPress ElasticSearch 架构
友情链接
  • Blogs·CN
  • Honesty
  • Mr.Sun的博客
  • 临窗旋墨
  • 哥斯拉
  • 彬红茶日记
  • 志文工作室
  • 懋和道人
  • 搬砖日记
  • 旧时繁华
  • 林羽凡
  • 瓦匠个人小站
  • 皮皮社
  • 知向前端
  • 蜗牛工作室
  • 韩小韩博客
  • 风渡言

COPYRIGHT © 2026 lifengdi.com. ALL RIGHTS RESERVED.

域名年龄

Theme Kratos Made By Dylan

津ICP备2024022503号-3

京公网安备11011502039375号