如何在没有Springockito的情况下模拟Spring bean

我在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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/360940.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

WP8手机上的图标

一直不清楚WP8手机上两个圆的标志是什么意思,今天看到下面的链接,终于搞明白了,原来是打开了GPS就有。 http://www.windowsphone.com/en-us/how-to/wp8/basics/what-do-the-icons-on-my-phone-mean 转载于:https://www.cnblogs.com/wonderow/…

JS保留两位小数

toFixed(2)

OpenShift上的Java EE工作流(技术提示#64)

该网络研讨会展示了如何使用WildFly , JBoss Tools , Forge , Arquillian和OpenShift在OpenShift上创建Java EE工作流。 具体来说,它谈论: 如何使用JBoss Developer Studio轻松开发Java EE应用程序并将其直接部署到Op…

获取css样式

js模块代码 function getStyle(obj,attr){ if(obj.currentStyle){//兼容ie return obj.currentStyle[attr]; } else{ return getComputedStyle(obj,null)[attr]; } } window.οnlοadfunction(){ oDiv document.getElementsById(div); console.log((oDiv,width)); } HTML <d…

具有CompletableFuture的异步超时

有一天&#xff0c;我重写了执行不佳的多线程代码&#xff0c;该代码在Future.get()某个时刻被阻塞&#xff1a; public void serve() throws InterruptedException, ExecutionException, TimeoutException {final Future<Response> responseFuture asyncCode();final …

js 中对于 css 的变量操作(React也可)

文章目录前言一、设置CSS变量&#xff1f;二、读取变量三、删除变量总结前言 主要讲js 中对于 css 的变量操作&#xff1b; 前端框架&#xff1a;antd框架 一、设置CSS变量&#xff1f; document.body.style.setProperty(--primary, #7F583F’);二、读取变量 document.body…

ASIHTTPRequest类库简介和使用说明

官方网站&#xff1a; http://allseeing-i.com/ASIHTTPRequest/ 。可以从上面下载到最新源码&#xff0c;以及获取到相关的资料。 使用iOS SDK中的HTTP网络请求API&#xff0c;相当的复杂&#xff0c;调用很繁琐&#xff0c;ASIHTTPRequest就是一个对CFNetwork API进行了封装&a…

UltraESB的首选IDE – IntelliJ IDEA

在AdroitLogic&#xff0c;我们长期以来一直在使用IntelliJ IDEA进行开发。 它是Java和相关语言/技术的最佳IDE&#xff08;它可能也是许多其他语言的选择&#xff0c;但我的经验主要是Java和相关技术&#xff09;。 Groovy和IDEA的Grails的集成很棒。 通过自动发现JDBC驱动程…

跟我一步一步开发自己的Openfire插件

这篇是简单插件开发&#xff0c;下篇聊天记录插件。 开发环境&#xff1a; System&#xff1a;Windows WebBrowser&#xff1a;IE6、Firefox3 JavaEE Server&#xff1a;tomcat5.0.2.8、tomcat6 IDE&#xff1a;eclipse、MyEclipse 8开发依赖库&#xff1a; Jdk1.6、jasper-com…

React div加载背景图

backgroundImage: url(${ActivitiesBack})

Apache FOP与Eclipse和OSGi的集成

Apache FOP是由XSL格式化对象&#xff08; XSL-FO &#xff09;驱动的开源打印处理器。 例如&#xff0c;将数据对象转换为PDF可能非常有用。 但是&#xff0c;将其集成到PDE中并最终以OSGi Service的形式运行并最终显得有些麻烦。 因此&#xff0c;我提供了一个P2存储库&…

预览视频

window.URL.createObjectURL(files[0].file),

C++ 四

//运算符函数&#xff0c; 重载。 #include<iostream> using namespace std; class A{ int data; public: A(int d0):data(d){ } void show(){ cout<<"data1"<<data<<endl; } //friend A operator-(const A& a1,const A& a2);//授权…

不删除侦听器–使用ListenerHandles

听一个可观察的实例并对它的变化做出反应很有趣。 做一些必要的事情来打断或结束这种聆听会变得很有趣。 让我们看看问题的根源和解决方法。 总览 这篇文章将首先讨论这种情况&#xff0c;然后再讨论常见的方法和问题所在。 然后&#xff0c;它将提供解决大多数问题的简单抽象…

将url参数字符串转成数组

const url"/BaseDictionary?Type34"; const arrurl.split(?); // arr["/BaseDictionary","Type34"]; typeStr parse(arr[1]); // typeStr{Type: "34"}

uniapp /deep/设置uni-app组件样式时 h5生效 小程序失效问题解决

今天写uni-app的项目 设置uni-app扩展组件的样式 使用穿透/deep/ 发现小程序没有效果 h5有效果 //小程序无效 h5生效 /deep/ .uni-list-item .uni-list-item__container .uni-list-item__content .uni-list-item__content-title{color: #333333;font-size: 32upx;}加入一下代…

使用Google Guava Cache进行本地缓存

很多时候&#xff0c;我们将不得不从数据库或另一个Web服务获取数据或从文件系统加载数据。 在涉及网络呼叫的情况下&#xff0c;将存在固有的网络等待时间&#xff0c;网络带宽限制。 解决此问题的方法之一是在应用程序本地拥有一个缓存。 如果您的应用程序跨越多个节点&…

uva 1394poj 3517

递推&#xff0c;把问题转化为具有相同问题的子问题&#xff0c;通过子问题最后所剩余的编号&#xff0c;退出此问题所剩余的编号 #include <iostream> using namespace std; const int maxn1000010; int f[maxn]; int main() {int n,k,m;while(~scanf("%d %d %d&qu…

父级和子级div的点击事件相互影响

解决方法&#xff1a;event.stopPropagation();

Jersey和Spring Boot入门

除了许多新功能&#xff0c;Spring Boot 1.2还带来了Jersey支持。 这是吸引喜欢标准方法的开发人员的重要一步&#xff0c;因为他们现在可以使用JAX-RS规范构建RESTful API&#xff0c;并将其轻松部署到Tomcat或任何其他Springs Boot支持的容器中。 带有Spring平台的Jersey可以…