我一直在尝试使用mockito和junit运行以下测试,并继续获取“java.lang.NullPointerException:name must not null” 谁能告诉我为什么会这样?
在调试时,我发现当测试在isStopValid(String)方法中执行以下语句时抛出此异常:
FacilityValidationUtil facUtil = new FacilityValidationUtil();
@RunWith(MockitoJUnitRunner.class)
public class MyFormTest{
@InjectMocks MyForm form = new MyForm();
@Mock FacilityValidationUtil facUtil;
@Test
public void testIsStopValid() throws FinderException{
when(facUtil.isFacilityValid("")).thenReturn(false);
form.setOrigin("");
assertEquals(false, form.isStopValid(form.getOrigin()));
}
}
包含待测方法的类:
public class MyForm{
FacilityValidationUtil facUtil = new FacilityValidationUtil();
public boolean isStopValid(String stop){
try {
return facUtil.isFacilityValid(stop);
} catch (FinderException e) {
log.error("Error finding the stop. "+e.getCause());
return false;
}
}
}
public class FacilityValidationUtil{
private FacilityDAO facilityDao = new HibernateFacilityDAO();
public boolean isFacilityValid(String facility) throws FinderException{
boolean test;
FacilityImpl facilityImpl = facilityDao.findFacilityByNassCode(facility);
test = (facilityImpl==null)?false : true;
return test;
}
}
public class HibernateFacilityDAO extends HibernateAbstractDeltaDAO implements FacilityDAO {
public HibernateFacilityDAO() {
super(false);
}
}
答案 0 :(得分:2)
简答:您正在尝试模拟isStopValid方法本地的变量(facUtil),因此您的测试中此对象的模拟版本永远不会被调用,因为您每次都在“新建它” 。
长答案:看起来你正试图模拟对FacilityValidationUtil类的调用,如果是这种情况,那么你需要让类成为一个字段,以便Mockito可以通过反射注入对象(如果这样对象是线程安全的,它看起来像是)或探索像PowerMockito这样的模拟框架,它允许你模拟一个构造函数(google for PowerMockito)。
PowerMockito.whenNew(FacilityValidationUtil.class).withNoArguments().thenReturn(facUtil);
默认情况下,Mockito不支持任何构造函数args的模拟。
修改强> 如果你仍然遇到麻烦,那么我建议从一个较小的例子开始。我已经为你设置了一个可以工作并使用你想要测试的代码(虽然它使用的是内部类,Mockito有一些古怪的规则,但我只是这样做来压缩这个例子)。
@RunWith(MockitoJUnitRunner.class)
public class MyFormTest {
@InjectMocks
private MyForm form = new MyForm();
@Mock
private FacilityValidationUtil facUtil;
@Test
public void testIsStopValid_false() {
when(facUtil.isFacilityValid("")).thenReturn(false);
assertEquals(false, form.isStopValid(""));
}
@Test
public void testIsStopValid_true() {
when(facUtil.isFacilityValid("")).thenReturn(true);
assertEquals(true, form.isStopValid(""));
}
public class MyForm {
private FacilityValidationUtil facUtil = new FacilityValidationUtil();
public boolean isStopValid(String stop) {
try {
return facUtil.isFacilityValid(stop);
} catch (FinderException e) {
return false;
}
}
}
public class FacilityValidationUtil {
public boolean isFacilityValid(String facility) throws FinderException {
throw new RuntimeException(facility);
}
}
public class FinderException extends RuntimeException {
public FinderException(String message) {
super(message);
}
}
}
真正重要的是你的模拟没有正确注入。在你解决之前,你将继续得到同样的错误。在您调用facUtil.isFaciltyValid的位置设置MyForm中的断点并查看该对象。它应该是一个模拟对象,而不是你的类。
祝你好运。