如何对在测试对象内部创建的模拟对象产生期望?

Posted

技术标签:

【中文标题】如何对在测试对象内部创建的模拟对象产生期望?【英文标题】:How to make expectations on a mock object created inside the tested object? 【发布时间】:2015-03-22 18:40:11 【问题描述】:

我想对一个如下所示的类进行单元测试:

template <typename T>
class MyClass 
...
    void someMethod() 
        T object;
        object.doSomething();
    
...
;

我想对这个类进行单元测试,所以我为 T 创建了一个模拟类:

struct MockT 
...
    MOCK_METHOD(doSomething, 0, void());
...
;

那我想在一个测试用例中使用它:

BOOST_AUTO_TEST_CASE(testSomeMethod) 
    MyClass<MockT> myClassUnderTest;
    MOCK_EXPECT(???)....;
    myClassUnderTest.someMethod();

我如何对这个对象做出期望?我的第一个想法是将所有创建的MockT 实例存储在构造函数的静态容器中,然后从析构函数的容器中删除它们。如果对象的创建方法与使用它的方法不同,这将起作用,如下所示:

myClassUnderTest.createTheObject();
MOCK_EXPECT(MockT::findMyObject().doSomething);
myClassUnderTest.useTheObject();

但是为此我需要修改我的类的接口,我真的不想这样做。还有什么我可以做的吗?

【问题讨论】:

【参考方案1】:

如果您不想修改接口或引入额外的间接性,可以使用 Typemock Isolator++。

template <typename T>
class MyClass

public:
    void someMethod()
    
        T object;
        object.doSomething();
    
;

    class RealType  //RealType is the actual production type, no injection needed
    
    public:
        void doSomething()
    ;

由于 T 是在 someMethod 内部(被测方法内部)创建的,我们需要伪造 T 的 ctor。 FAKE_ALL 就是这样做的。 fakeRealType 上设置的行为将适用于在运行时创建的所有 RealType 实例。默认的 FAKE_ALL 行为是递归伪造,这意味着所有伪造的方法都是伪造的,并将返回伪造的对象。您还可以在任何方法上手动设置您想要的任何行为。

TEST_CLASS(MyTests)
    
    public:

        TEST_METHOD(Faking_Dependency_And_Asserting_It_Was_Called)
        
            RealType* fakeRealType= FAKE_ALL<RealType>();
            MyClass<RealType> myClassUnderTest;
            myClassUnderTest.someMethod();

            ASSERT_WAS_CALLED(fakeRealType->doSomething()); 
        

    ;

Typemock fakes 并不严格,所以你需要编写一个适当的断言来确保你的方法确实被调用了。您可以使用 Typemock 提供的 ASSERT_WAS_CALLED 来完成。

P.S 我用的是 MSTest。

【讨论】:

【参考方案2】:

您可以将 doSomething 成员函数重定向到 static one,例如

struct MockT

    void doSomething() 
        soSomethingS();
    
    MOCK_STATIC_FUNCTION( doSomethingS, 0, void(), doSomething )
;

那么你的测试将是

BOOST_AUTO_TEST_CASE(testSomeMethod) 
    MyClass<MockT> myClassUnderTest;
    MOCK_EXPECT(MockT::doSomething).once();
    myClassUnderTest.someMethod();

如果需要,您可以测试对象实例的construction 和destruction,但它可能不会为您的测试带来更多。

【讨论】:

是否可以从模拟构造函数的操作中获取this?因为那时也许我可以在对象被创建的时候对它做出期望。 不是来自构造函数的动作,它与构造函数具有相同的签名,因此此处为 void(),但如果需要,可以重复相同的习语,例如让构造函数仅充当存根转发到 MOCK_STATIC_FUNCTION 传递“this”。【参考方案3】:

我发现最好的方法是为成员使用共享指针。不幸的是,我不得不因为单元测试而使用额外的间接,但至少它工作得很好。

template <typename T, typename TFactory>
class MyClass 
...
    void someMethod() 
        std::shared_ptr<T> object = factory();
        object->doSomething();
    
...
    TFactory factory;
;

然后在测试中它看起来像这样:

BOOST_AUTO_TEST_CASE(testSomeMethod) 
    std::shared_ptr<T> mockT;
    MockTFactory factory(mockT);
    MyClass<MockT, MockTFactory> myClassUnderTest(factory);
    MOCK_EXPECT(mockT->doSomething).once();
    myClassUnderTest.someMethod();

【讨论】:

以上是关于如何对在测试对象内部创建的模拟对象产生期望?的主要内容,如果未能解决你的问题,请参考以下文章

如何在 javascript 中模拟内部 JSON 对象?

使用spock在被测试函数内部使用的模拟对象

浅谈面向对象之封装继承多态!如何使用内部类模拟多继承

Java基础——面向对象(内部类)

对象内部套嵌多个对象

面向对象之内部类