0

我是嘲笑的新手,所以我正在玩这个场景。我有一个MyClass.process()调用MyClass.getFactory(). 我想模拟getFactory()失败,以便我可以确保process()正确恢复。

似乎如果我getFactory()在模拟之后调用它会正确抛出异常,但如果它是通过原始process()方法调用的,则不会抛出异常。

public class MyClass {

    public MyFactory getFactory() throws FactoryException {
        return ...
    }

    public String process() {
        try {
            MyFactory factory = getFactory();
            ...
        }
        catch(FactoryException ex) {
            ...
        }
    }

}

测试类:

public class MyClassTest {

    @Test
    public void testGetFactoryFails() {
        try {
            MyClass mockMyClass = mock(MyClass.class);
            when(mockMyClass.getFactory()).thenThrow(new FactoryException());
            mockMyClass.process();
            fail("FactoryException should have been thrown.");
        }
        catch(FactoryException ex) {
            // all good
        }
    }

}

感谢所有的反馈。我可以看到在更复杂的环境下模拟单个方法会很快变得难以管理。getFactory() 的内容并不重要,因为这是概念代码,而不是生产代码。

我尝试过的一种方法效果很好:

public class MyClassTest {

    class MyClassMock1 extends MyClass {

        @Override
        public MyFactory getFactory() throws FactoryException {
            throw new FactoryException("Bad factory instance.");
        }

    }

    @Test
    public void testGetFactoryFails() {
        try {
            MyClassMock1 mockMyClass = new MyClassMock1();
            mockMyClass.process();
            fail("FactoryException should have been thrown.");
        }
        catch(FactoryException ex) {
            assertEquals("Bad factory instance.", ex.getMessage());
        }
    }

}
4

2 回答 2

1

我将扩展@Can't Tell已经说过的内容。您使用模拟库来模拟被测类的依赖关系。在您的情况下,您进行测试MyClass,并且在测试期间您应该模拟所有依赖项(MyClass使用的类)。

您想在一个类中模拟一个方法并测试另一种方法。虽然这在技术上可能是可行的(但由于各种代理/包装机制对您不起作用),但它肯定是一个糟糕的设计。

告诉我们你的getFactory()方法。如果它单独处理整个工作,请将其提取到单独的类并模拟它。如果它只委托给另一个类,则模拟该类。可悲的是,这个提取的类将被命名为MyFactoryFactory......

提取依赖项后,对其进行模拟并MyClass在测试之前传递给它(通过构造函数、设置器...)

于 2012-09-30T14:51:36.723 回答
0

这是因为部分嘲笑是不好的(这就是你想要做的)。它恰恰导致了这类问题。它主要用于遗留系统测试。如果您编写新代码并且必须使用部分模拟,这意味着您很可能有糟糕的系统设计。

您应该getFactory将调用委托给其他服务(注入MyClass),然后您应该模拟整个服务

于 2012-09-30T14:51:56.070 回答