我在Spring工作了几年。 但是我总是对XML配置变得多么混乱感到沮丧。 随着各种注释和Java配置可能性的出现,我开始喜欢使用Spring进行编程。 这就是为什么我强烈使用Java配置的原因。 我认为,仅当您需要可视化Spring Integration或Spring Batch流时,XML配置才适用。 希望Spring Tool Suite还将能够可视化这些框架的Java配置。

XML配置的令人讨厌的方面之一是,它通常会导致庞大的XML配置文件。 因此,开发人员经常创建用于集成测试的测试上下文配置。 但是,如果不对生产布线进行测试,那么集成测试的目的是什么? 这样的集成测试几乎没有价值。 因此,我一直试图以可测试的方式设计生产环境。

我除了在创建新项目/模块时会尽可能避免XML配置。 因此,使用Java配置,您可以为每个模块/包创建Spring配置,并在主上下文中对其进行扫描(@Configuration也是组件扫描的候选者)。 这样,您可以自然地创建岛屿Spring bean。 这些岛可以很容易地进行隔离测试。

但是我必须承认,并非总是可以按原样测试生产Java配置。 很少需要修改某些豆的行为或监视。 有一个名为Springockito的库。 老实说,到目前为止,我还没有使用过它,因为我总是尝试设计Spring配置以避免进行模拟。 纵观Springockito的发展速度和未解决的问题数量 ,我有点担心将其引入我的测试套件堆栈中。 实际上,最后一个发行版是在Spring 4发行版之前完成的,带来了诸如“是否可以轻松地将其与Spring 4集成?”之类的问题。 我不知道,因为我没有尝试过。 如果需要在集成测试中模拟Spring bean,我更喜欢纯Spring方法。

Spring提供了@Primary批注,用于指定在注册了两个相同类型的bean时首选哪个bean。 这很方便,因为您可以在集成测试中用伪造的Bean覆盖生产Bean。 让我们探索这种方法,并举例说明一些陷阱。

我选择了这种简单/虚拟的生产代码结构进行演示:

@Repository
public class AddressDao {public String readAddress(String userName) {return "3 Dark Corner";}
}@Service
public class AddressService {private AddressDao addressDao;@Autowiredpublic AddressService(AddressDao addressDao) {this.addressDao = addressDao;}public String getAddressForUser(String userName){return addressDao.readAddress(userName);}
}@Service
public class UserService {private AddressService addressService;@Autowiredpublic UserService(AddressService addressService) {this.addressService = addressService;}public String getUserDetails(String userName){String address = addressService.getAddressForUser(userName);return String.format("User %s, %s", userName, address);}
}

AddressDao单例bean实例注入到AddressServiceAddressServiceUserService也类似使用。

我必须在此阶段警告您。 我的方法对生产代码略有侵入。 为了能够伪造现有的生产Bean,我们必须在集成测试中注册伪造的Bean。 但是这些假bean通常与生产bean位于同一包子树中(假设您使用的是标准Maven文件结构:“ src / main / java”和“ src / test / java”)。 因此,当它们在同一包子树中时,将在集成测试期间对其进行扫描。 但是我们不想在所有集成测试中都使用所有bean的假货。 伪造品可能会破坏无关的集成测试。 因此,我们需要一种机制,如何告诉测试仅使用某些假豆。 这是通过从组件扫描中完全排除假豆来完成的。 集成测试明确定义了正在使用的伪造品(稍后将显示)。 现在让我们看一下从组件扫描中排除假豆的机制。 我们定义了自己的标记注释:

public @interface BeanMock {
}

并在主要的Spring配置中从组件扫描中排除@BeanMock批注。

@Configuration
@ComponentScan(excludeFilters = @Filter(BeanMock.class))
@EnableAutoConfiguration
public class Application {
}

组件扫描的根包是Application类的当前包。 因此,所有上述生产Bean都必须位于同一包装或子包装中。 现在,我们需要为UserService创建集成测试。 让我们窥探地址服务bean。 当然,使用此生产代码进行此类测试没有实际意义,但这仅是示例。 这是我们的间谍豆:

@Configuration
@BeanMock
public class AddressServiceSpy {@Bean@Primarypublic AddressService registerAddressServiceSpy(AddressService addressService) {return spy(addressService);}
}

生产AddressService bean是从生产上下文自动连接的,包装到Mockito的间谍中,并注册为AddressService类型的主bean。 @Primary批注确保我们的假bean将用于集成测试而不是生产bean。 @BeanMock批注可确保Application组件扫描无法扫描此bean。 现在让我们看一下集成测试:

@RunWith(SpringJUnit4ClassRunner.class)
@SpringApplicationConfiguration(classes = { Application.class, AddressServiceSpy.class })
public class UserServiceITest {@Autowiredprivate UserService userService;@Autowiredprivate AddressService addressService;@Testpublic void testGetUserDetails() {// GIVEN - spring context defined by Application class// WHENString actualUserDetails = userService.getUserDetails("john");// THENAssert.assertEquals("User john, 3 Dark Corner", actualUserDetails);verify(addressService, times(1)).getAddressForUser("john");}
}

@SpringApplicationConfigration批注具有两个参数。 首先( Application.class )声明受测的Spring配置。 第二个参数( AddressServiceSpy.class )指定将用于测试的假bean加载到Spring IoC容器中。 显然,我们可以根据需要使用尽可能多的bean伪造品, 但是您不想拥有太多的bean伪造品。 这种方法应该很少使用,如果您经常观察自己使用这种模拟,那么您的应用程序或开发团队中的紧密耦合可能会遇到严重的问题。 TDD方法论应该可以帮助您解决此问题。 请记住:“减少嘲笑总是更好!”。 因此,请考虑进行生产设计更改,以减少模拟的使用。 这也适用于单元测试。

在集成测试中,我们可以自动连接此间谍bean并将其用于各种验证。 在这种情况下,我们验证了测试方法userService.getUserDetails addressService.getAddressForUser使用参数“ john”调用了方法addressService.getAddressForUser

我再举一个例子。 在这种情况下,我们不会监视生产bean。 我们将模拟它:

@Configuration
@BeanMock
public class AddressDaoMock {@Bean@Primarypublic AddressDao registerAddressDaoMock() {return mock(AddressDao.class);}
}

再次,我们重写了生产bean,但是这次我们用Mockito的模拟代替它。 然后,我们可以在集成测试中记录模拟行为:

@RunWith(SpringJUnit4ClassRunner.class)
@SpringApplicationConfiguration(classes = { Application.class, AddressDaoMock.class })
public class AddressServiceITest {@Autowiredprivate AddressService addressService;@Autowiredprivate AddressDao addressDao;@Testpublic void testGetAddressForUser() {// GIVENwhen(addressDao.readAddress("john")).thenReturn("5 Bright Corner");// WHENString actualAddress = addressService.getAddressForUser("john");// THENAssert.assertEquals("5 Bright Corner", actualAddress);}@Afterpublic void resetMock() {reset(addressDao);}
}

我们通过@SpringApplicationConfiguration的参数加载@SpringApplicationConfiguration的bean。 在测试方法中,当将“ john”作为参数传递给它时,将存根addressDao.readAddress方法以返回“ 5 Bright Corner”字符串。

但是请记住,记录的行为可以通过Spring上下文进行不同的集成测试。 我们不希望测试相互影响。 因此,您可以通过在测试后重置模拟来避免测试套件中将来出现问题。 这是在方法resetMock完成的。

  • 源代码在Github上 。

翻译自: https://www.javacodegeeks.com/2014/12/how-to-mock-spring-bean-without-springockito.html

如何在没有Springockito的情况下模拟Spring bean相关推荐

  1. spring-bean版本_如何模拟Spring bean(版本2)

    spring-bean版本 大约一年前,我写了一篇博客文章如何模拟Spring Bean . 所描述的模式对生产代码几乎没有侵入性. 正如读者Colin在评论中正确指出的那样,基于@Profile注释 ...

  2. 如何模拟Spring bean(版本2)

    大约一年前,我写了一篇博客文章如何模拟Spring Bean . 所描述的模式对生产代码几乎没有侵入性. 正如读者Colin在评论中正确指出的那样,基于@Profile注释的间谍/模拟Spring b ...

  3. 在没有XML的情况下测试Spring和Hibernate

    我非常热衷于Spring 3中的改进,这些改进最终使您能够在IDE和编译器的适当支持下从XML迁移到纯Java配置. 它并没有改变Spring是一个庞大的套件这一事实,并且有时发现您需要的东西可能需要 ...

  4. 在无需分叉的情况下模拟任何 SIGHASH 标志

    我们开发了一种新颖的方法来模拟任何 SIGHASH 标志,只需在智能合约中编写逻辑即可.它不需要更改协议,因此比每次构思新用例时通过分叉添加硬编码标志更实用和灵活. SIGHASH 标志 SIGHAS ...

  5. python在材料模拟中的应用_材料模拟python_模拟-python模拟-在不妨碍实现的情况下修补方法...

    此答案解决了Quuxplusone用户提供的赏金中提到的其他要求: 对于我的用例而言,重要的是它可以与MagicMock一起使用,即,它不需要我在构造Potato(在此示例中为spud)实例与调用sp ...

  6. 编写函数模拟掷骰子的游戏(两个骰子)。第一次掷的时候,如果点数之和为7或11则获胜;如果点数之和为2、3或12则落败;其他情况下的点数之和称为“目标”,游戏继续。在后续的投掷中,如果玩家再次掷出“目标

    编写函数模拟掷骰子的游戏(两个骰子).第一次掷的时候,如果点数之和为7或11则获胜:如果点数之和为2.3或12则落败:其他情况下的点数之和称为"目标",游戏继续.在后续的投掷中,如 ...

  7. linux 磁盘延时,Linux:如何模拟硬盘延迟?我想在不使用CPU电源的情况下增加iowait值...

    设备映射器"延迟"设备 查看设备映射器设备的"延迟"目标.这正是它存在的原因. 例 这是如何进行此操作的示例: 创建一个可以读取/写入的位置 [root@cen ...

  8. java模拟浏览器不关闭会话_JSP实现浏览器关闭cookies情况下的会话管理

    通常,会话管理是通过服务器将 Session ID 作为一个 cookie 存储在用户的 Web 浏览器中来唯一标识每个用户会话.如果浏览器不支持 cookies,或者将浏览器设置为不接受 cooki ...

  9. 今年,自动驾驶卡车将在无人驾驶的情况下上路

    文章来源:IEEE电气电子工程师 Photo: TuSimple First in Freight: In 2021, San Diego–based startup TuSimple plans t ...

最新文章

  1. SAP MCH1表和MCHA表更新逻辑
  2. Quartz.Net 调度框架配置介绍
  3. vim编辑环境设定[Linux]
  4. Java 中正确获取中文字符串长度
  5. HP LaserJet 1010卡纸解决方法
  6. 用 cctld工具创建带有国家代码的IP地址表
  7. web前端之HTML
  8. 深入解析:DBA_OBJECTS中的OBJECT_ID与DATA_OBJECT_ID的区别
  9. 教你 7 步快速构建 GitLab 持续集成环境
  10. Java语言程序设计(沈泽刚主编)第3版 第1~3章编程练习答案
  11. 暗时间 pdf_时间管理最全的一篇:理论+实践+电子书
  12. ICMAX解析运行内存发展新趋势 LPDDR4X将会给手机带来哪些改变?
  13. Excel键盘快捷键大全(二)
  14. xp怎样修改计算机mac地址,xp系统怎么修改mac地址
  15. 微信小程序获取openid(用户唯一身份识别)
  16. Linux挂载Windows共享文件夹
  17. Catia V5-6R2016安装教程
  18. java计算机毕业设计五金机电市场批发零售管理信息系统源程序+mysql+系统+lw文档+远程调试
  19. php 设定title,PHP中title是什么意思,title如何编辑!
  20. ElGamal加密算法简介

热门文章

  1. 2019蓝桥杯省赛---java---A---6(完全二叉树的权值)
  2. mysql group和order_mysql 用 group by 和 order by同时使用
  3. 3-7 基于SpringBoot的Apache Shiro环境快速搭建与配置实操
  4. python下面的代码_解析一下下面的python代码?
  5. datagridview绑定数据源不显示_sharding-jdbc系列之 数据源配置(一)
  6. java.util.concurrent.locks.Lock文档说明
  7. 请求nginx静态资源报403
  8. redis的主从数据库复制功能
  9. java开发指南_Java 12新功能完整指南
  10. 因此,Oracle杀死了java.net