spring boot 单元测试JUnit5使用Mockito模拟Mock数据调用

这篇具有很好参考价值的文章主要介绍了spring boot 单元测试JUnit5使用Mockito模拟Mock数据调用。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

spring boot 单元测试JUnit5使用Mockito模拟Mock数据调用

好大一批新用法,大家静下心来好好看看吧

1. spring boot 使用 Mockito.when().thenReturn()模拟返回值

Mockito 是一种 Java mock 框架,他主要就是用来做 mock 测试的,他可以模拟任何 Spring 管理的 bean、模拟方法的返回值、模拟抛出异常…等,他同时也会记录调用这些模拟方法的参数、调用顺序,从而可以校验出这个 mock 对象是否有被正确的顺序调用,以及按照期望的参数被调用

SpringBoot 目前内建的是 Mockito 框架

1)测试mockito.when设置固定值ID=1
/**
 * mockito.when设置固定值ID=1
 */
@Test
void mockito_1L() {
    User user = new User();
    user.setId(1L);
    // 配置userService的模拟行为,mock拦截了userService的真实行为,查询Id=1时,会模拟返回user
    Mockito.when(userService.selectById(1L)).thenReturn(user);
    JsonResult query = testUserController.query(user);
    Assertions.assertEquals(user, query.getData());
}
2)测试mockito.when_AnyLong 设置任意Long值
/**
 * 测试mockito.when_AnyLong 设置任意Long值
 */
@Test
void mockito_AnyLong() {
    User mockUser = new User();
    mockUser.setId(1L);
    mockUser.setUsername("我是mock用户名");
    // 配置userService的模拟行为,mock拦截了userService的真实行为,查询任意ID时,会模拟返回user
    // 打桩,会拦截userService.selectById的方法
    Mockito.when(userService.selectById(Mockito.anyLong())).thenReturn(mockUser);

    // 查询ID=2L的
    User user1 = new User();
    user1.setId(1L);
    JsonResult query = testUserController.query(user1);
    Assertions.assertEquals(mockUser, query.getData());

    // 查询ID=2L的
    User user2 = new User();
    user2.setId(2L);
    JsonResult query2 = testUserController.query(user2);
    Assertions.assertEquals(mockUser, query2.getData());
    // 查询出来的结果,都是mockUser
}
3)测试mockito.when_AnyLong,测试调用多个参数的方法

当传入两个参数,其中一个参数采用任意参数时,指定参数需要matchers来对比,比如Mcokito.eq(“123456”)文章来源地址https://www.toymoban.com/news/detail-553074.html

/**
 * 测试mockito.when_AnyLong,测试调用多个参数的方法
 */
@Test
void mockito_AnyLong2() {
    User mockUser = new User();
    mockUser.setId(1L);
    mockUser.setUsername("我是mock用户名");
    mockUser.setPassword("123456");
    // 当传入两个参数,其中一个参数采用任意参数时,指定参数需要matchers来对比
    // 当传入任意用户名字符串,和123456密码时,返回mock用户,否则返回null
    Mockito.when(userService.addUser2Params(Mockito.anyString(), Mockito.eq("123456")))
        .thenReturn(mockUser);

    // controller去调用
    JsonResult jsonResult = testUserController
        .addUser2Params("xXxxXxx", mockUser.getPassword());

    User user = (User) jsonResult.getData();
    // 匹配上了,返回值与mockUser相同
    Assertions.assertEquals(mockUser, user);

    JsonResult jsonResult1 = testUserController.addUser2Params("xXxxXxx", "xX");

    User user1 = (User) jsonResult1.getData();
    // 不匹配,返回值null与mockUser就相同了
    Assertions.assertNotEquals(mockUser, user1);
}
4)测试mockito.when_any(User.class) 设置对象
/**
 * 测试mockito.when_any(User.class) 设置对象
 */
@Test
void mockito_AnyClass() {
    // 打桩,拦截userService.addUserSynchronizedGood方法,传入任意User对象,都会返回true
    Mockito.when(userService.addUserSynchronizedGood(Mockito.any(User.class)))
            .thenReturn(true);

    // 增加good1
    User user1 = new User();
    user1.setId(1L);
    user1.setUsername("good1");
    JsonResult add1 = testUserController.addUserSynchronizedGood(user1);
    Assertions.assertEquals(true, add1.getData());

    // 增加good2
    User user2 = new User();
    user2.setId(2L);
    user1.setUsername("good2");
    JsonResult add2 = testUserController.addUserSynchronizedGood(user2);
    Assertions.assertEquals(true, add2.getData());
    // 返回结果,都是mockUser
}
5)测试mockito.when_特定的对象的值
/**
 * 测试mockito.when_特定的对象的值
 */
@Test
void mockito_Object() {
    User params = new User();
    params.setUsername("good1");

    // 一旦一个参数是任意匹配,其他参数都必须是任意匹配
    // 打桩,拦截userService.addUserSynchronizedGood方法,传入good1对象,都会返回true
    // Mockito.when(userService.addUserSynchronizedGood(Mockito.eq(params))).thenReturn(true);
    Mockito.when(userService.addUserSynchronizedGood(params)).thenReturn(true);

    // 增加good1
    User user1 = new User();
    user1.setUsername("good1");
    JsonResult add1 = testUserController.addUserSynchronizedGood(user1);
    Assertions.assertEquals(true, add1.getData());

    // 增加good2
    User user2 = new User();
    user2.setUsername("good2");
    JsonResult add2 = testUserController.addUserSynchronizedGood(user2);
    Assertions.assertNotEquals(true, add2.getData());
    // 返回结果,都是mockUser
}
2. spring boot 使用mockito.when_模拟返回异常
1)thenThrow 模拟有返回值方法的异常返回
/**
 * mockito.when_thenReturn 模拟返回异常
 */
@Test
void mockito_thenReturn() {
    User params = new User();
    params.setUsername("bad1");

    // 当bad1时,模拟抛出异常
    Mockito.when(userService.addUserSynchronizedGood(params))
            .thenThrow(new RuntimeException("bad1异常"));

    // 增加good1
    User user1 = new User();
    user1.setUsername("good1");
    JsonResult add1 = testUserController.addUserSynchronizedGood(user1);
    Assertions.assertEquals(false, add1.getData());

    // 增加bad1
    User user2 = new User();
    user2.setUsername("bad1");
    Assertions.assertThrows(RuntimeException.class,
            () -> testUserController.addUserSynchronizedGood(user2), "错误了,没有产生异常");
}
2)doThrow 模拟没有返回值方法的异常返回
/**
 * mockito.doThrow().when 没有返回值的方法模拟返回异常
 */
@Test
void mockito_toThrow() {
    User params = new User();
    params.setUsername("bad");

    // 当userService.addUser(params),值是bad时,会模拟抛出异常
    Mockito.doThrow(new RuntimeException("没有返回值的bad异常"))
            .when(userService).addUser(params);

    // 增加bad,会有异常
    User user1 = new User();
    user1.setUsername("bad");
    Assertions.assertThrows(RuntimeException.class,
            () -> testUserController.addUser(user1), "错误了,没有产生异常");

    // 增加good,不会触发异常
    User user2 = new User();
    user2.setUsername("good");
    Assertions.assertDoesNotThrow(() -> 
            testUserController.addUserSynchronizedGood(user2), "错误了,产生异常了");
}
3. spring boot 使用 mockito_verify 校验验证
1) mockito_verify 校验验证本次测试调用过哪些方法,调用了多少次
/**
 * mockito_verify 校验验证本次测试调用过哪些方法,调用了多少次
 */
@Test
void mockito_verify() {
    userService.selectById(3L);
    userService.selectById(3L);
    userService.selectById(3L);
    // 验证selectById是否调用了3次
    Mockito.verify(userService, Mockito.times(3)).selectById(Mockito.eq(3L));

    User user = new User();
    user.setId(3L);
    testUserController.query(user);
    testUserController.query(user);
    testUserController.query(user);
    // 验证selectById是否调用了6次
    Mockito.verify(userService, Mockito.times(6)).selectById(Mockito.eq(3L));
}
2) mockito.inOrder.verify,按照顺序校验
/**
 * mockito.inOrder.verify,按照顺序校验
 */
@Test
void mockito_inOrder() {
    userService.selectById(1L);
    userService.selectById(3L);
    userService.selectById(2L);
    // 按顺序校验
    InOrder inOrder = Mockito.inOrder(userService);
    // 校验1次1L
    inOrder.verify(userService).selectById(1L);
    // 之后调了1次3L
    inOrder.verify(userService).selectById(3L);
    // 之后又调了1次2L
    inOrder.verify(userService).selectById(2L);

}
4. 用到的TestUserController
package space.goldchen.springboot.test;

import lombok.RequiredArgsConstructor;
import org.springframework.web.bind.annotation.*;
import space.goldchen.springboot.common.JsonResult;
import space.goldchen.springboot.entity.User;
import space.goldchen.springboot.service.UserService;

/**
 * 单元测试,由单元测试来验证测试
 *
 * @author chenzhao
 * @create 2023-05-24 17:00
 */
@RestController
@RequestMapping("/test")
@RequiredArgsConstructor
public class TestUserController {
    private final UserService userService;

    /**
     * 新增用户
     *
     * @param user
     * @return
     */
    @PostMapping("/add")
    public JsonResult addUserSynchronizedGood(@RequestBody User user) {
        boolean b = userService.addUserSynchronizedGood(user);
        return new JsonResult(b);
    }

    /**
     * 新增用户,没有加锁的
     *
     * @param user
     * @return
     */
    @PostMapping("/addUser")
    public JsonResult addUser(@RequestBody User user) {
        userService.addUser(user);
        return new JsonResult("添加了用户");
    }

    /**
     * 查询用户
     *
     * @param user
     * @return
     */
    @GetMapping("/queryById")
    public JsonResult query(@RequestBody User user) {
        User user1 = userService.selectById(user.getId());
        return new JsonResult(user1);
    }

    /**
     * 查询用户
     *
     * @param id
     * @return
     */
    @GetMapping("/queryById2")
    public JsonResult query2(Long id) {
        User user1 = userService.selectById(id);
        return new JsonResult(user1);
    }


    /**
     * 修改用户
     *
     * @param user
     * @return
     */
    @PutMapping("/updateById")
    public JsonResult update(@RequestBody User user) {
        boolean b = userService.updateSelective(user);
        return new JsonResult(b);
    }


    /**
     * 删除用户
     *
     * @param user
     * @return
     */
    @DeleteMapping("/updateById")
    public JsonResult delete(@RequestBody User user) {
        boolean b = userService.del(user);
        return new JsonResult(b);
    }
}
5.用到的UserService
package space.goldchen.springboot.service;

import space.goldchen.springboot.entity.User;

/**
 * @author chenzhao
 * @create 2023-05-20 9:56
 */
public interface UserService {
    User selectByUsername(String username);

    User selectById(Long id);

    void addUser(User user);

    /**
     * 同步锁,添加用户
     * @param user
     */
    boolean addUserSynchronizedBad(User user);
    /**
     * 同步锁,添加用户
     * @param user
     */
    boolean addUserSynchronizedGood(User user);

    /**
     * 更新用户
     * @param user
     * @return
     */
    boolean updateSelective(User user);

    /**
     * 删除用户
     * @param user
     * @return
     */
    boolean del(User user);
}

到了这里,关于spring boot 单元测试JUnit5使用Mockito模拟Mock数据调用的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

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

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

相关文章

  • Junit5+Mockito单元测试详解

    1.宏观层面:AIR原则 A:Automatic(自动化) 全自动执行,输出结果无需人工检查,而是通过断言验证。 I:Independent(独立性) 分层测试,各层之间不相互依赖。 R:Repeatable(可重复) 可重复执行,不受外部环境( 网络、服务、中间件等)影响。 2.微观层面:BCDE原则 B: Bord

    2024年01月17日
    浏览(33)
  • springboot项目使用Junit5 + mockito + jacoco 实现单元测试以及代码覆盖率检查

    在创建springboot项目时会默认添加spring-boot-starter-test依赖,其中已经包含了junit、mockito依赖,根据springboot版本的不同junit和mockito的版本也会有所不同 先说一下各自功能: junit只说一点,junt4和junit5的注解不同,使用方式略有差异,其他不赘述了,基本用法都懂。 mockito是mock的

    2023年04月23日
    浏览(41)
  • 单元测试-spring-boot-starter-test+junit5

    开发过程中经常需要写单元测试,记录一下单元测试spring-boot-starter-test+junit5的使用 引用jar包 可以对局部函数、方法进行调用测试 可以对API调用进行模拟测试

    2024年01月19日
    浏览(35)
  • 深度揭秘JUnit5与Mockito的单元测试神秘面纱

    在今天的学习中,我们将深入研究 JUnit 和Mockito,这是 Java 开发中最强大的 单元测试 工具之一。通过学习如何编写清晰、高效的单元测试,我们将揭开单元测试的神秘面纱,助力你在项目中写出更健壮的代码。 提示: 今天的代码是在第九天代码的基础上进行开发,我们将为

    2024年02月02日
    浏览(40)
  • SpringBoot 实战:JUnit5+MockMvc+Mockito 做好单元测试

    因为继承了 spring-boot-starter-parent ,所以我们依赖的 spring-boot-starter-test 不需要写具体的版本,可以直接集成父级的版本定义。其中, spring-boot-starter-web 是用于提供 REST API 的 web 容器, spring-boot-starter-test 可以提供各种测试框架的, spring-boot-maven-plugin 是将 SpringBoot 应用打包为

    2024年04月15日
    浏览(38)
  • Spring Boot与微服务测试:JUnit和Mockito的单元和集成测试实践

    微服务架构的流行使得测试变得更为重要。在Spring Boot应用中,使用JUnit和Mockito进行单元和集成测试是一种常见的实践。本文将深入探讨如何利用这两个测试框架,确保Spring Boot微服务的可靠性和稳定性。 单元测试 1.1 JUnit简介 JUnit是Java中最为流行的单元测试框架之一,它提供

    2024年02月21日
    浏览(32)
  • spring boot 单元测试JUnit5使用MockMvc调用get请求,post请求,设置head请求头,解析返回值json和字符串

    spring boot 单元测试JUnit5使用MockMvc调用get/post接口 源码地址:https://gitcode.net/qq_39339588/springboot.git 1. 先准备一份controller,一会儿供测试调用 2. MockMvc测试调用get请求接口 两个注解说明 @SpringBootTest // 加测试类上,标明是测试的类 @AutoConfigureMockMvc // 支持对MockMvc对象的注入和配置,

    2024年02月14日
    浏览(38)
  • JUnit 5单元测试(三)—— Mockito 模拟

    上一篇讲了 JUnit 5单元测试(二)—— 断言,书接上文开始 JUnit 5单元测试(三)—— Mockito 模拟 想象下面这几种情况你该怎么单元测试: 1.A方法去数据库查询了数据进行了一些处理,该怎么单元测试; 2.在微服务项目中,A方法中调用了远程微服务B方法(或者B方法还没写好

    2024年02月15日
    浏览(37)
  • Spring Cloud中Controller单元测试 Junit5 & MockMvc

    在Spring Cloud中进行Controller的单元测试,使用Junit5和Mock。 Controller: 方式一:使用@SpringBootTest + @AutoConfigureMockMvc 方式二:使用@WebMvcTest + @ImportAutoConfiguration(RefreshAutoConfiguration.class) 解决 No Scope registered for scope name \\\'refresh\\\' 异常 注入Mockmvc方式有两种 方式一:(@AutoConfigureMockMvc / @

    2024年02月16日
    浏览(29)
  • 在Spring Boot环境中使用Mockito进行单元测试

    Mockito是一个流行的Java mocking框架,它允许开发者以简单直观的方式创建和使用模拟对象(mocks)。Mockito特别适用于在Spring Boot环境中进行单元测试,因为它能够轻松模拟Spring应用中的服务、存储库、客户端和其他组件。通过使用Mockito,开发者可以模拟外部依赖,从而使单元测

    2024年03月23日
    浏览(34)

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

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

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

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

二维码1

领取红包

二维码2

领红包