[英]How to run DefaultMessageListenerContainer on the 'main' thread
[英]How to mock DefaultMessageListenerContainer
我在模擬(使用Mockito) DefaultMessageListenerContainer
( org.springframework.jms.listener.DefaultMessageListenerContainer
)時遇到困難。 這是我的代碼:
@Mock
private DefaultMessageListenerContainer defaultMessageListenerContainer;
@Before
public void init() {
MockitoAnnotations.initMocks( this );
incomingFeedController = new IncomingFeedControllerImpl();
}
@Test
public void testHandleConnectionState() {
List< DefaultMessageListenerContainer > listeners =
new ArrayList< DefaultMessageListenerContainer >();
listeners.add( defaultMessageListenerContainer );
incomingFeedController.setContainers( listeners );
when( defaultMessageListenerContainer.isRunning() ).thenReturn( false );
}
然后,我想做一些適當的測試,例如:
when( defaultMessageListenerContainer.isRunning() ).thenReturn( false );
但是在junit運行之后,此行最終顯示為:
java.lang.NullPointerException at org.springframework.jms.listener.AbstractJmsListeningContainer.isRunning(AbstractJmsListeningContainer.java:312) at com.source.etf.manager.integrationgateway.feedcontroller.IncomingFeedControllerTest.testHandleConnectionState(IncomingFeedControllerTest.java:37) at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at sun.reflect.NativeMethodAccessorImpl.invoke(Unknown Source) at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source) at java.lang.reflect.Method.invoke(Unknown Source) at org.junit.internal.runners.TestMethod.invoke(TestMethod.java:59) at org.junit.internal.runners.MethodRoadie.runTestMethod(MethodRoadie.java:98) at org.junit.internal.runners.MethodRoadie$2.run(MethodRoadie.java:79) at org.junit.internal.runners.MethodRoadie.runBeforesThenTestThenAfters(MethodRoadie.java:87) at org.junit.internal.runners.MethodRoadie.runTest(MethodRoadie.java:77) at org.junit.internal.runners.MethodRoadie.run(MethodRoadie.java:42) at org.junit.internal.runners.JUnit4ClassRunner.invokeTestMethod(JUnit4ClassRunner.java:88) at org.junit.internal.runners.JUnit4ClassRunner.runMethods(JUnit4ClassRunner.java:51) at org.junit.internal.runners.JUnit4ClassRunner$1.run(JUnit4ClassRunner.java:44) at org.junit.internal.runners.ClassRoadie.runUnprotected(ClassRoadie.java:27) at org.junit.internal.runners.ClassRoadie.runProtected(ClassRoadie.java:37) at org.junit.internal.runners.JUnit4ClassRunner.run(JUnit4ClassRunner.java:42) at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:50) at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38) at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:467) at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:683) at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:390) at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:197)
我還檢查了AbstractJmsListeningContainer
,這是發生NPE的代碼:
public final boolean isRunning() {
synchronized (this.lifecycleMonitor) {
return (this.running && runningAllowed());
}
}
我發現的是lifecycleMonitor
對象沒有被實例化。 此對象在頂部的AbstractJmsListeningContainer
聲明:
protected final Object lifecycleMonitor = new Object();
任何想法如何正確模擬DefaultMessageListenerContainer
嗎?
您仍然需要在要初始化的對象上設置模擬。
我看到您正在測試IncomingFeedControllerImpl
,並且您的模擬對象很可能是此類的實例的成員。 因為您沒有在IncomingFeedControllerImpl
顯式設置模擬的DefaultMessageListenerContainer
,所以AbstractJmsListeningContainer
(可能是@Autowired
)仍在徘徊,並且未被模擬。
您將需要使用setter方法或構造函數來注入它。 (您也可以@Autowired
它)
Mockito無法模擬最終類或最終方法。 這些約束由JVM本身強制執行。 模擬這樣的代碼實際上需要重寫類字節碼並將其加載到另一個類加載器中。 這導致非常復雜的代碼。 PowerMock朝那個方向發展,並且代碼難以維護。
另外,請勿嘲笑您不擁有的類型,請參閱google上的4-5首結果。 為什么需要在單元測試中模擬Spring類型。 您應該創建一些間接操作以避免Spring遵從性,或者編寫集成測試。 后者似乎更合適,因為它與JMS有關。
private DefaultMessageListenerContainer createFailingContainer() {
DefaultMessageListenerContainer container = new DefaultMessageListenerContainer() {
@Override
public void start() throws JmsException {
throw new MyJmsException("TEST");
}
@Override
public void stop() throws JmsException {
throw new MyJmsException("TEST");
}
};
return container;
}
class MyJmsException extends JmsException {
private static final long serialVersionUID = 1L;
public MyJmsException(String msg) {
super(msg);
}
}
您要么需要使用PowerMock,要么按照@Brice的建議進行集成測試。 這是在PowerMock中執行的方法:
@RunWith(PowerMockRunner.class)
@PrepareForTest(DefaultMessageListenerContainer.class)
public class MyTestClass {
// cannot use the @Mock annotation
private DefaultMessageListenerContainer defaultMessageListenerContainer;
@Before
public void init() {
// This should allow the mocking of final methods as well.
defaultMessageListenerContainer =
PowerMockito.mock(DefaultMessageListenerContainer.class);
incomingFeedController = new IncomingFeedControllerImpl();
}
@Test
public void testHandleConnectionState() {
List< DefaultMessageListenerContainer > listeners =
new ArrayList< DefaultMessageListenerContainer >();
listeners.add( defaultMessageListenerContainer );
incomingFeedController.setContainers( listeners );
when( defaultMessageListenerContainer.isRunning() ).thenReturn( false );
}
}
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.