如何在JUnit和Spring中测试void返回方法的java最佳实践?
我不确定我是否以正确的方式测试了void
返回方法,也不确定我的class-under-test
(cut)是否需要任何更改以使其100%可测试和防bug
我在执行test
时看到NullPointerException
,因为loginOperations
没有设置
错误:
java.lang.NullPointerException
at com.demo.service.LoginService.doLogin(LoginService.java:40)
at com.demo.service.LoginServiceTest.doLogin(LoginServiceTest.java:25)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:498)
at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47)
at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17)
at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:78)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:57)
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290)
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58)
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268)
at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
at org.junit.runner.JUnitCore.run(JUnitCore.java:137)
at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:68)
at com.intellij.rt.execution.junit.IdeaTestRunner$Repeater.startRunnerWithArgs(IdeaTestRunner.java:47)
at com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:242)
at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:70)
登录服务。java
@Service
public class LoginService {
@Autowired
private ILoginOperations loginOperations;
public void doLogin(HttpServletRequest request, String result) {
LoginDTO loginDTO = new LoginDTO(request.getParameter("username"), result);
loginOperations.doLogin(loginDTO);
}
}
登录服务测试。java
public class LoginServiceTest {
private LoginService instance = new LoginService();
ILoginOperations loginOperations = Mockito.mock(ILoginOperations.class);
HttpServletRequest request = Mockito.mock(HttpServletRequest.class);
String result = "some string";
@Test
public void doLogin() {
when(request.getParameter("username")).thenReturn("johndoe");
instance.doLogin(request, result); //throws NPE while calling loginOperations.doLogin() because
assertNotNull(instance); //IS THIS THE CORRECT WAY TO TEST A VOID RETURNING METHOD ???
}
}
现在,有两种方法可以修复测试
- 我可以通过为
loginOperations
类添加setter
方法并在test
中调用setter
方法来修复被测试的类 - 将
@Test public void doLogin() {
更改为@Test(expected = Exception.class) public void doLogin() {
不确定上面哪一个是最佳实践以及原因
另一个问题:
我的另一个问题是如何对一个不返回任何内容的方法执行assert
。有类似verify()
的东西,但不确定如何使用它
# 1 楼答案
一,。您可以通过在
LoginService
中添加setter方法来修复测试用例,也可以使用构造函数注入,比如-@Test(expected = Exception.class) public void doLogin()
当然不是一个好主意,因为doLogin方法在正常情况下不会抛出异常李>使用void返回类型测试方法的更好方法是使用验证API(示例-mockito verification API example)。您还可以使用Mockito的ArgumentCaptor捕获参数并断言该参数的状态,以及验证API,如下所示:
Martin Fowler有一篇关于这种测试方法的优秀文章——Mocks Aren't Stubs
# 2 楼答案
实际上,你应该为你的LoginService创建一个构造函数来获取ILoginOperations,这样你就可以在测试类中创建LoginService,并将模拟的ILoginOperations作为参数传递,所有这些都应该在@Before方法中完成
或者你可以尝试使用@InjectMocks作为你的登录服务,并将你的ILoginOperations注释为@Mock