如何在方法内部测试匿名类对象

use*_*853 2 java unit-testing anonymous-class

我正在使用testNg和Mockito编写单元案例。我面临的问题是以下给出的方法:

public void publishRequest(final NotificationRequest request){

    MessageCreator messageCreator = new MessageCreator() {

        @Override
        public Message createMessage(Session session) throws JMSException {
            if (logger.isDebugEnabled()) {
                logger.debug("Start of MessagePublisher publishRequest");
            }
            ObjectMessage msg = session.createObjectMessage();

            msg.setStringProperty(
                    "SELECTOR",
                    request.getAvailaibilityTier() + "-"
                            + request.getEsbReference());

            if (logger.isDebugEnabled()) {
            logger.debug("Message Selector=>"+msg.getStringProperty("SELECTOR"));
            }

            msg.setStringProperty(NotificationConstants.CACHE_KEY.name(),
                    request.getId());

            msg.setObject(request.getData());
            if (logger.isDebugEnabled()) {
                logger.debug("Publishing request -->" + request);
            }

            if (logger.isDebugEnabled()) {
                logger.debug("End of MessagePublisher publishRequest");
            }
            return msg;
        }
    };
    producerTemplate.send(messageCreator);
}
Run Code Online (Sandbox Code Playgroud)

在编写单元案例时,即使我的测试案例创建了MessageCreater对象,也不会调用createMessage()方法。我想在单位案例中也包括在内,以获得更好的覆盖范围。任何想法如何测试内部类的方法?

Nil*_*lsH 5

可以说,内部内部类是该publishRequest方法的实现细节。在进行单元测试时,通过测试合同publishRequest而非内部实现,您将获得更强大的测试。正如@vtheron建议的那样,要测试您的MessageCreator,您必须将其提取到自己的类中。但是除此之外,还应该考虑MessageCreator在课堂上“注入”对您的引用。这样,您就可以同时测试publishRequestMessageCreator隔离。就个人而言,我更喜欢在构造函数中“注入”这样的依赖关系,但是还有其他方法可以实现相同的依赖关系。我会做这样的事情:

public class MyClass {
    private MessageCreator messageCreator;

    public MyClass(MessageCreator messageCreator) {
        this.messageCreator = messageCreator;
    }

    public void publishRequest(final NotificationRequest request) {
        producerTemplate.send(messageCreator);
    }
}
Run Code Online (Sandbox Code Playgroud)

然后,您只需MessageCreator在测试中向您的班级发送“模拟” 即可:

public class MyTest {

    @Test
    public void testPublish() {
        MessageCreator mock = new MessageCreator() { // Mock impl (or use a mock library)};
        MyClass classToTest = new MyClass(mock);
        classToTest.publishRequest(notificationRequest);
        // Do assertions and verify
    }
}
Run Code Online (Sandbox Code Playgroud)

它仍在测试publishRequest方法的内部(测试void方法趋向于测试),但是如果您可以像

发布请求时,应将消息发送到队列

而不是

publishRequest应该创建一个MessageCreator并创建一个由生产者发送的消息

至少您有一种更加抽象的编写测试的方法,从长远来看,从定义上来讲,这将影响您编写测试的方式。