单元测试实战(四)MyBatis-Plus 的测试

这篇具有很好参考价值的文章主要介绍了单元测试实战(四)MyBatis-Plus 的测试。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

为鼓励单元测试,特分门别类示例各种组件的测试代码并进行解说,供开发人员参考。

本文中的测试均基于JUnit5。

单元测试实战(一)Controller 的测试

单元测试实战(二)Service 的测试    

单元测试实战(三)JPA 的测试

单元测试实战(四)MyBatis-Plus 的测试

单元测试实战(五)普通类的测试

单元测试实战(六)其它

概述

MyBatis Plus组件表现为Mapper对象(我们将不涉及IService的测试)。使用MyBatis/MyBatis-Plus的项目,往往有很多自写的SQL需要测试。

MyBatis Plus有专门的@MyBatisPlusTest注解,是苞米豆提供的功能,它是有Spring上下文的,使用JUnit的SpringExtension扩展类。与@DataJpaTest一样,它会在测试时自动将数据源替换为内存数据库的。

在本章的示例中,我们将展示一种使用SQL脚本来准备测试数据的方法。

断言应主要检查数据存取行为是否符合预期。

依赖

除依赖Spring自带测试框架外,还需要苞米豆的测试包以及内存数据库:

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-test</artifactId>
    <scope>test</scope>
</dependency>
<dependency>
    <groupId>org.junit.jupiter</groupId>
    <artifactId>junit-jupiter-api</artifactId>
    <scope>test</scope>
</dependency>
<dependency>
    <groupId>com.baomidou</groupId>
    <artifactId>mybatis-plus-boot-starter-test</artifactId>
    <version>3.5.3.2</version>
    <scope>test</scope>
</dependency>
<dependency>
    <groupId>com.h2database</groupId>
    <artifactId>h2</artifactId>
    <scope>test</scope>
</dependency>

注意:mybatis-plus-boot-starter-test的3.5.2版本支持Spring Boot 2,3.5.3及以上支持Spring Boot 3。

示例

由于篇幅的关系,本章中我们将拿出一个Mapper方法作为示例。

以下为xml实现的一个查询方法:

<select id="selectPermissionCodesByUser" resultType="java.lang.String">
    SELECT p.code
    FROM sys_auth_user_role ur
        JOIN sys_auth_role_permission rp ON rp.role_id=ur.role_id
        JOIN sys_auth_permission p ON p.id=rp.permission_id
    WHERE p.type=2 AND ur.user_code=#{userCode} AND ur.asset_type=#{assetType} AND ur.asset_id=#{assetId}
 
    <if test="pkOrg != null">
        UNION DISTINCT
 
        SELECT p.code
        FROM sys_auth_user_org uo
            JOIN sys_auth_role_permission rp ON rp.role_id=uo.role_id
            JOIN sys_auth_permission p ON p.id=rp.permission_id
        WHERE p.type=2 AND uo.user_code=#{userCode} AND uo.pk_org=#{pkOrg}
    </if>
</select>

以下为该查询的代理接口:

public interface SysAuthPermissionMapper extends BaseMapper<SysAuthPermission> {
    ...
    Set<String> selectPermissionCodesByUser(@Param("userCode") String userCode,
                                            @Param("assetType") int assetType,
                                            @Param("assetId") long assetId,
                                            @Param("pkOrg") String pkOrg);
    ...
}

以下是该查询方法的测试类:

package com.aaa.sdk.rbac.mybatis.database.mapper;
 
import com.baomidou.mybatisplus.test.autoconfigure.MybatisPlusTest;
import org.junit.jupiter.api.BeforeAll;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.autoconfigure.jdbc.AutoConfigureTestDatabase;
import org.springframework.core.io.ClassPathResource;
import org.springframework.jdbc.datasource.init.ScriptUtils;
import org.springframework.test.context.TestPropertySource;
 
import javax.sql.DataSource;
import java.sql.Connection;
import java.util.Set;
 
import static org.assertj.core.api.Assertions.assertThat;
 
@MybatisPlusTest
@AutoConfigureTestDatabase(replace = AutoConfigureTestDatabase.Replace.NONE)
@TestPropertySource(properties = {
        "spring.datasource.driver-class-name=org.h2.Driver",
        "spring.datasource.url=jdbc:h2:mem:aaa_rbac_test;DB_CLOSE_DELAY=-1;DB_CLOSE_ON_EXIT=FALSE;MODE=MySQL;DATABASE_TO_LOWER=TRUE",
        "spring.datasource.username=sa"
})
class SysAuthPermissionMapperTest {
 
    @Autowired
    private SysAuthPermissionMapper mapper;
 
    @BeforeAll
    static void setupClass(@Autowired DataSource dataSource) throws Exception {
        try (Connection conn = dataSource.getConnection()) {
            ScriptUtils.executeSqlScript(conn, new ClassPathResource("/sql/test_schema.sql"));
            ScriptUtils.executeSqlScript(conn, new ClassPathResource("/sql/test_data.sql"));
        }
    }
 
    @Test
    void testSelectPermissionCodesByUser() {
        Set<String> permissions = mapper.selectPermissionCodesByUser("wangfei012", 1, 1, "0001A410000000A3I0V2");
        assertThat(permissions).hasSize(4);
    }
 
    @Test
    void testSelectPermissionCodesByUser_emptyPkOrg() {
        Set<String> permissions = mapper.selectPermissionCodesByUser("wangfei012", 1, 1, "");
        assertThat(permissions).hasSize(3);
    }
 
    @Test
    void testSelectPermissionCodesByUser_invalidUser() {
        Set<String> permissions = mapper.selectPermissionCodesByUser("nobody", 1, 1, "0001A410000000A3I0V2");
        assertThat(permissions).hasSize(0);
    }
}

测试类说明:

第18行,我们使用了@MyBatisPlusTest类注解。

第19行,为了H2数据库兼容MySQL语法,我们关闭了数据源的自动替换,选择用20-24行的代码来覆盖默认的数据源。

第28行,我们将代理接口注入测试类。

31行的setupClass方法会在所有测试方法执行之前执行一次,准备数据。这里我们用了Spring自带的ScriptUtils实用程序;其中sql脚本是在test目录下的resources文件夹;测试目录结构如下图:

mybatisplus 单元测试,单元测试,MyBatis-Plus测试,怎么测mapper

上图中,MockApplication与测试在同一个包内,用来控制Spring上下文范围,以免启动真实的SpringBootApplication;它的代码如下(注意里面有个@MapperScan,用来发现当前包里的MyBatis mapper):

package com.aaa.sdk.rbac.mybatis.database.mapper;
 
import org.mybatis.spring.annotation.MapperScan;
import org.springframework.boot.autoconfigure.SpringBootApplication;
 
 
@SpringBootApplication
@MapperScan(basePackages = {"com.aaa.sdk.rbac.mybatis.database.mapper"})
public class MockApplication {
}

以下继续测试类的说明:

从第38行开始,是selectPermissionCodesByUser方法的三个测试用例。

testSelectPermissionCodesByUser测试正常流程,包括全部4个正常参数。按照设计和测试数据,它应该取出4个结果。

testSelectPermissionCodesByUser_emptyPkOrg测试传入的pkOrg参数为空时的场景,覆盖的是xml文件中<if test="pkOrg != null">不满足的分支。按照设计和测试数据,它应该取出3个结果。

testSelectPermissionCodesByUser_invalidUser则是一个负面测试,测试当传入的userCode非法时的场景;它应该不返回任何结果。

与JPA的测试一样,测试类中没有Mock对象,因此也就不存在given - when - then三段式结构。(DAO对象没有更底层的依赖,因此不需注入Mock,这也使得它们看上去更像是一种与数据库的集成测试。)

总结

使用@MybatisPlusTest。

为了兼容MySQL语法而使用@AutoConfigureTestDatabase(replace = AutoConfigureTestDatabase.Replace.NONE),并用@TestPropertySource订制数据源属性。注意:兼容只是部分兼容,并非完全兼容。

使用@BeforeAll方法结合Spring自带的ScriptUtils准备测试数据。

MockApplication里加@MapperScan。

示例中,我们没有在每个方法之前重置测试数据,这是因为我们只测了查询方法吗?不。在DAO的测试中,不管是@MyBatisPlusTest还是@DataJpaTest,都带有@Transactional元注解,它们会在测试方法执行结束后自动回滚。

MyBatis-Plus 有所谓的 Lambda Query,可链式组合查询。我们不建议在Service的实现类里直接使用它;应该将其封装到DAO/Repository层的方法里,这样就方便mock这个封装好的方法的行为,避免直接mock链式组合查询行为。如果实在要mock链式组合查询行为,可以参考这篇和这篇。 文章来源地址https://www.toymoban.com/news/detail-791951.html

到了这里,关于单元测试实战(四)MyBatis-Plus 的测试的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • MyBatis-Plus 引入依赖导致的Error creating bean和java.lang.NoClassDefFoundError: com/baomidou/mybatisplus问题

     使用mybatis-plus出现的一些问题记录。 问题错误截图 问题原因: 依赖冲突 项目主pom文件中引入了mybatis-plus-boot-start依赖,版本为 3.5.3 业务包中引入mybatis-plus代码生成器依赖版本为 3.4.1 导致mybatis-plus-core核心包存在两个版本,导致冲突   解决办法 修改业务包中mybatis-plus代码

    2024年02月09日
    浏览(35)
  • Mybatis-plus怎么更新Null字段?

    本文介绍【Mybatis-plus】updateById()方法不能更新字段为null的原因及解决办法。 在日常项目开发过程中,经常会使用Mybatis-plus的updateById()方法,快速将接收道德参数或者查询结果中原本不为null的字段更新为null,并且该字段在数据库中可为null,这个时候使用updateById()并不能实现

    2024年02月13日
    浏览(29)
  • MyBatis-Plus 实战教程四 idea插件

    MybatisPlus提供了很多的插件功能,进一步拓展其功能。目前已有的插件有: PaginationInnerInterceptor:自动分页 TenantLineInnerInterceptor:多租户 DynamicTableNameInnerInterceptor:动态表名 OptimisticLockerInnerInterceptor:乐观锁 IllegalSQLInnerInterceptor:sql 性能规范 BlockAttackInnerInterceptor:防止全表更

    2024年02月06日
    浏览(31)
  • 【重要】springboot实战(六)之mybatis-plus代码自动生成器

    目录 环境: 步骤: 1.添加依赖 2.配置代码 3.运行 测试 1.测试生成的service 1.1、service用法 2.分页查询 2.1、分页插件配置  2.2、测试 3.源码 jdk:1.8 springboot版本:2.7.15 mybatis-plus版本:3.5.1以上 (本文章用的当前最新版本:3.5.3.2,代码适用于3.5.1版本以上的版本) 在测试类中创建

    2024年02月03日
    浏览(34)
  • 【Mybatis-Plus】Mybatis-Plus快速入门

    Mybatis-Plus是基于Mybatis的数据库操作组件,其实现的功能完全是Mybatis的功能拓展,不改变Mybatis的使用方式,可以兼容Mybatis的操作方式。 创建一个数据库、一个表进行基础操作: 创建一个Spring项目,项目通过Spring Initlizer创建,不导入任何依赖包,在POM.xml文件中进行依赖导入

    2024年02月07日
    浏览(36)
  • Mybatis-Plus 进阶开发 -- Mybatis-Plus 入门教程(二)

    为了巩固所学的知识,作者尝试着开始发布一些学习笔记类的博客,方便日后回顾。当然,如果能帮到一些萌新进行新技术的学习那也是极好的。作者菜菜一枚,文章中如果有记录错误,欢迎读者朋友们批评指正。 (博客的参考源码可以在我主页的资源里找到,如果在学习的

    2024年02月10日
    浏览(37)
  • Mybatis-Plus(三)--Mybatis-Plus配置和条件构造器

    在MP中有大量的配置,其中有一部分是Mybatis原生的配置,另一部分是MP的配置,详情:https://mybatis.plus/config 【1】configLocation--自己单独的MyBatis配置的路径 SpringMVC的xml中写法: 【2】mapperLocations--MyBatis Mapper所对应的XML文件位置 如果你在Mapper中有自定义方法(XML中有自定义实现

    2024年02月15日
    浏览(38)
  • Mybatis-Plus通用枚举功能 [MyBatis-Plus系列] - 第493篇

    历史文章( 文章 累计490+) 《国内最全的Spring Boot系列之一》 《国内最全的Spring Boot系列之二》 《

    2024年02月08日
    浏览(31)
  • mybatis-plus分页total为0,分页失效,mybatis-plus多租户插件使用

    背景:项目使用mybatis分页插件不生效,以及多租户使用时读取配置异常 多租户插件使用遇到的问题: 最开始在MyTenantLineHandler中使用 @Value(\\\"${tables}\\\"),服务启动时能从配置中心拉取到配置,但在运行时获取到的值为空,试了很多方法都不生效,后面将配置中心的配置在调用My

    2024年02月06日
    浏览(37)
  • Mybatis-Plus

    官方网站     MyBatis-Plus(简称 MP)是一个 MyBatis的增强工具,在 MyBatis 的基础上只做增强不做改变,为简化开发、提高效率而生。我们的愿景是成为 MyBatis 最好的搭档,就像 魂斗罗 中的 1P、2P,基友搭配,效率翻倍。   无侵入 :只做增强不做改变,引入它不会对现有工程产

    2024年01月15日
    浏览(26)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包