我一直在尝试使用嘲笑和junit运行以下测试,并且继续获取“ java.lang.NullPointerException:名称不能为null”
谁能告诉我为什么会这样吗?

在调试时,我发现测试在isStopValid(String)方法中执行以下语句时会引发此异常:

FacilityValidationUtil facUtil =新的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);
    }
}

最佳答案

简短的答案:您正在尝试模拟isStopValid方法本地的变量(facUtil),因此永远不会调用此对象在测试中的模拟版本,因为您每次都在“对其进行更新”。

长答案:看来您正在尝试模拟对FacilityValidationUtil类的调用,如果是这种情况,则需要将类设置为字段,以便Mockito可以通过反射注入对象(如果该对象是线程)安全性(看起来像是安全的)或探索类似PowerMockito的模拟框架,该框架可让您模拟构造函数(新时使用Google替代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中设置一个断点,然后查看对象。它应该是模拟对象,而不是您的类。

祝好运。

10-06 11:21