Mockito& Junit空指针异常:name不能为null

时间:2013-03-11 17:38:01

标签: java unit-testing junit tdd mockito

我一直在尝试使用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);
    }
}

1 个答案:

答案 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中的断点并查看该对象。它应该是一个模拟对象,而不是你的类。

祝你好运。