java - 使用 Mockito 的 Java Mock final void 方法
问题描述
我需要一些帮助来理解如何对以下内容进行单元测试。我有以下类结构。我必须按原样遵循结构,因为这是我们产品的框架,我不能偏离这一点。
package com.rohan.base
import javax.inject.Inject;
public abstract class ServiceBase {
@Inject
protected CommandExecutorBase commandExecutor;
public final void execute() {
commandExecutor.execute(this);
}
}
--------------------------------------------------------
package com.rohan.services
public class MyService extends ServiceBase {
public void someMethod() {}
}
--------------------------------------------------------
package com.rohan.delegates
import javax.inject.Inject;
import com.rohan.services.MyService
public class MyDelegate {
@Inject
MyService myService;
public void callService() {
myService.execute();
}
}
我正在尝试使用 TDD 方法进行编码,并且我想确保callService
委托类中的方法调用execute
Service 类上的方法。
我尝试了以下方法,但它在执行方法中给了我一个空指针异常(因为commandExecutor
对象为空)。有人可以告诉我我该怎么做吗?我不允许使用 PowerMockito。
import org.junit.Test;
import org.mockito.Mockito;
public class MyDelegateUnitTests {
@Test
public void validateServiceExecuteIsCalled() {
MyDelegate delegate = new MyDelegate();
MyService serviceObject = Mockito.mock(MyService.class);
delegate.myService = serviceObject;
Mockito.doAnswer((i) -> {
System.out.println("I am here");
return null;
}).when(serviceObject).execute();
Mockito.verify(serviceObject, Mockito.times(1)).execute();
}
}
我无法模拟 commandExecutor 对象,因为它不可访问(与我的 Delegate 类不同的包)。代替Mockito.doAnswer()
,我也试过Mockito.doNothing().when(serviceObject).execute();
但我得到了同样的例外。有人可以告诉我,当我试图在when
方法中定义一个模拟时,为什么它会继续尝试执行真正的方法?谢谢!
更新 我修改了测试类,并添加了文件夹/文件 mockito-extensions (正如@jokster 在下面的答案中所建议的那样)。但是,我最终遇到了以下异常。我正在使用 Mockito 2。
我的测试班:
import org.junit.Before;
import org.junit.Test;
import org.mockito.Mockito;
public class MyDelegateUnitTests {
MyDelegate delegate;
MyService serviceObject;
@Before
public void setUpMocks() {
delegate = new MyDelegate();
serviceObject = Mockito.mock(MyService.class);
Mockito.doNothing().when(serviceObject).execute();
delegate.myService = serviceObject;
}
@Test
public void validateSettersAreCalled() {
delegate.callService();
Mockito.verify(serviceObject, Mockito.times(1)).execute();
}
}
我得到以下异常:
java.lang.ExceptionInInitializerError: null
at org.mockito.internal.configuration.plugins.Plugins.getStackTraceCleanerProvider(Plugins.java:17)
at org.mockito.internal.exceptions.stacktrace.StackTraceFilter.<clinit>(StackTraceFilter.java:21)
at org.mockito.internal.exceptions.stacktrace.ConditionalStackTraceFilter.<init>(ConditionalStackTraceFilter.java:17)
at org.mockito.exceptions.base.MockitoException.filterStackTrace(MockitoException.java:41)
at org.mockito.exceptions.base.MockitoException.<init>(MockitoException.java:30)
at org.mockito.exceptions.misusing.MockitoConfigurationException.<init>(MockitoConfigurationException.java:18)
at org.mockito.internal.configuration.plugins.PluginLoader.loadImpl(PluginLoader.java:66)
at org.mockito.internal.configuration.plugins.PluginLoader.loadPlugin(PluginLoader.java:24)
at org.mockito.internal.configuration.plugins.PluginRegistry.<init>(PluginRegistry.java:12)
at org.mockito.internal.configuration.plugins.Plugins.<clinit>(Plugins.java:11)
at org.mockito.internal.util.MockUtil.<clinit>(MockUtil.java:24)
at org.mockito.internal.MockitoCore.<init>(MockitoCore.java:44)
at org.mockito.Mockito.<clinit>(Mockito.java:975)
at com.rohan.delegates.MyDelegateUnitTests.setUpMocks(MyDelegateUnitTests.java:16)
解决方案
作为单元测试的一部分,我认为不需要定义myService.execute
方法的行为。定义行为是不必要的
Mockito.doAnswer((i) -> {
System.out.println("I am here");
return null;
}).when(serviceObject).execute();
您可以删除上面的代码。
验证该execute
方法是否以正确的次数被调用就足够了。要回答您的问题,一个建议是与基于构造函数的注入有关。MyDelegate
基本上为和注入创建一个构造函数Myservice
。在测试中,执行以下操作:
MyService serviceObject = Mockito.mock(MyService.class);
MyDelegate delegate = new MyDelegate(serviceObject);
如果您不想在委托类中使用基于构造函数的注入,则其他选择是在测试中使用@InjectMocks
和。@Inject
推荐阅读
- reactjs - 手动安装 React 应用程序的最佳方法
- mongodb - MongoTemplate 有没有办法返回子文档键
- node.js - NodeJS 等效于 curl 请求
- sql-server - 如何在 SQL Server 中获取表的表定义
- c# - 主键自增
- algorithm - 板尺寸大于 64 的位板算法?
- gatling - Gatling2 未能正确使用用户会话
- c++ - 为什么这个三角形条只画了屏幕的三分之一?
- apache-flink - 数据流入之前的所有源准备就绪 - 在整个 Flink 作业/数据流中
- python - Tensorflow - 使用parallel_interleave从多个tfrecord中读取不同的block_lengths?