我使用Java 11,JUnit Jupiter 5.6.0,Mockito 3.3.3和Java Discord API 4.1.1_122为小型Discord机器人编写了一些测试用例类。

我意识到我的测试用例具有多个重复行为的实例,包括用于相同目的的相等模拟的声明,以及在我的given([method call]).willReturn([object]);启动方法中重复@BeforeEach的情况。我希望将重复的行为分成基类来组织自己,但我被困住了。
我所有的测试类都在Java Discord API框架中测试某种事件处理程序,因此我制作了一个名为GenericEventTest的顶级类:

public abstract class GenericEventTest {
    @Mock
    protected GenericEvent eventMock;

    @Mock
    protected JDA jdaMock;

    protected GenericEventTest() {
        MockitoAnnotations.initMocks(this);

        given(   eventMock.getJDA()   )
                .willReturn(jdaMock);
    }
}


我希望为该事件实现一个高级模拟,以使其多态适应我的测试类,然后将指定的given()命令放入其所属的类中,因为我的所有测试都重复同一行。
接下来,我意识到我所有的测试都在测试与Discord中消息相关的某种事件。所以我扩展了GenericEvent并制成了GuildMessageEventTest

public abstract class GuildMessageEventTest extends GenericEventTest {
    @Mock
    protected GenericGuildMessageEvent eventMock;

    @Mock
    protected TextChannel eventChannelMock;

    @Mock
    protected MessageAction messageActionMock;

    protected GuildMessageEventTest() {
        super();

        MockitoAnnotations.initMocks(this);

        given(   eventMock.getChannel()   )
                .willReturn(eventChannelMock);

        given(   eventChannelMock.sendMessage(anyString())   )
                .willReturn(messageActionMock);
    }
}


我的想法是在此类中实现下一个事件级别,即GenericGuildMessageEvent,它将对given()应用相同的GenericEvent规则,然后在此类中添加适当的given()规则。但是我相信我已经找到了问题。

当我声明@Mock protected GenericGuildMessageEvent eventMock;时,我在GenericEvent eventMock中遮盖了GenericEventTest。因此,我声明了GenericGuildMessageEvent实例,但是我没有保留given()中附加到模拟程序的GenericEventTest规则。不能执行@Mock protected GenericGuildMessageEvent eventMock = (GenericGuildMessageEvent) super.eventMock;之类的操作,因为用@Mock注释的对象不能以这种方式合法地转换。

我的问题是:在让Mockito复制附加到超类中GenericEvent实例的GenericGuildMessageEvent规则的同时,如何合法地将given()的实例上传到GenericEvent

最佳答案

abstract GenericEvent getEventMock();中声明一个GenericEventTest方法,并让GuildMessageEventTest实现它。然后将实例字段protected GenericEvent eventMock从基类移动到子类。使用getEventMock()访问基类中的模拟。请注意,您的子类中不再需要MockitoAnnotations.initMocks(this);。这将覆盖基类的模拟行为。

// GenericEventTest
import org.mockito.Mock;
import org.mockito.MockitoAnnotations;

import static org.mockito.BDDMockito.given;

public abstract class GenericEventTest {

    protected GenericEventTest() {
        MockitoAnnotations.initMocks(this);

        given(getEventMock().getJDA()).willReturn("jdaMock");
    }

    abstract GenericEvent getEventMock();
}

// GuildMessageEventTest
import org.mockito.Mock;
import org.mockito.MockitoAnnotations;

import static org.mockito.BDDMockito.given;

public abstract class GuildMessageEventTest extends GenericEventTest {
    @Mock
    protected GenericGuildMessageEvent eventMock;

    @Mock
    protected GenericGuildMessageEvent myMock;

    @Override
    public GenericGuildMessageEvent getEventMock() {
        return eventMock;
    }

    protected GuildMessageEventTest() {
        super();

//        MockitoAnnotations.initMocks(this);

        given(eventMock.getChannel())
                .willReturn("eventChannelMock");

    }
}

//The Test
import org.junit.jupiter.api.Test;

import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertNotNull;

public class ActualMessageTest extends GuildMessageEventTest {

    @org.junit.jupiter.api.Test
    public void testX() {
        assertEquals("eventChannelMock", eventMock.getChannel());
        assertEquals("jdaMock", eventMock.getJDA());
        assertNotNull(myMock);
    }

}

关于java - 有没有一种方法可以合法地转换Mockito模拟对象?,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/60871568/

10-09 03:30