JMS 配置中可能出现的奇怪行为 Class
Possible Strange Behaviour in JMS Config Class
这是我之前 post 此处的一个单独但相关的问题:
我使用 spring 启动的 JMS 应用程序正确处理了所有内容,并且没有错误地关闭。为了让它工作,我从以下位置更改了一个 bean:
@Bean
public DefaultJmsListenerContainerFactory jmsListenerContainerFactory() {
DefaultJmsListenerContainerFactory factory = new DefaultJmsListenerContainerFactory();
factory.setConnectionFactory(mqConnectionFactory());
factory.setDestinationResolver(destinationResolver());
factory.setConcurrency("1");
factory.setErrorHandler(errorHandler());
factory.setSessionTransacted(true);
factory.setSessionAcknowledgeMode(Session.CLIENT_ACKNOWLEDGE);
return factory;
}
收件人:
@Bean
public DefaultMessageListenerContainer defaultMessageListenerContainer() {
DefaultMessageListenerContainer jmsListenerContainer = new DefaultMessageListenerContainer();
jmsListenerContainer.setConnectionFactory(mqConnectionFactory());
jmsListenerContainer.setDestinationResolver(destinationResolver());
jmsListenerContainer.setDestinationName(queueName);
jmsListenerContainer.setConcurrency("1");
jmsListenerContainer.setErrorHandler(errorHandler());
jmsListenerContainer.setSessionTransacted(true);
jmsListenerContainer.setSessionAcknowledgeMode(Session.CLIENT_ACKNOWLEDGE);
jmsListenerContainer.setAutoStartup(false);
return jmsListenerContainer;
}
问题是,我本可以只创建一个 "hotfix",因为我对 spring 的了解很少。当我偶然发现这个 post 时添加了更改的 bean jmsListenerContainer.setAutoStartup(false);
中的行: http://forum.spring.io/forum/spring-projects/integration/79176-illegalstateexception-no-message-listener-specified 因为没有 autoStartup
设置为 false 我在日志中的每条处理消息后得到这个:
java.lang.IllegalStateException: No message listener specified - see property 'messageListener'
at org.springframework.jms.listener.AbstractMessageListenerContainer.invokeListener(AbstractMessageListenerContainer.java:691) [spring-jms-4.2.4.RELEASE.jar:4.2.4.RELEASE]
at org.springframework.jms.listener.AbstractMessageListenerContainer.doExecuteListener(AbstractMessageListenerContainer.java:651) [spring-jms-4.2.4.RELEASE.jar:4.2.4.RELEASE]
at org.springframework.jms.listener.AbstractPollingMessageListenerContainer.doReceiveAndExecute(AbstractPollingMessageListenerContainer.java:315) [spring-jms-4.2.4.RELEASE.jar:4.2.4.RELEASE]
at org.springframework.jms.listener.AbstractPollingMessageListenerContainer.receiveAndExecute(AbstractPollingMessageListenerContainer.java:253) [spring-jms-4.2.4.RELEASE.jar:4.2.4.RELEASE]
at org.springframework.jms.listener.DefaultMessageListenerContainer$AsyncMessageListenerInvoker.invokeListener(DefaultMessageListenerContainer.java:1150) [spring-jms-4.2.4.RELEASE.jar:4.2.4.RELEASE]
at org.springframework.jms.listener.DefaultMessageListenerContainer$AsyncMessageListenerInvoker.executeOngoingLoop(DefaultMessageListenerContainer.java:1142) [spring-jms-4.2.4.RELEASE.jar:4.2.4.RELEASE]
at org.springframework.jms.listener.DefaultMessageListenerContainer$AsyncMessageListenerInvoker.run(DefaultMessageListenerContainer.java:1039) [spring-jms-4.2.4.RELEASE.jar:4.2.4.RELEASE]
at java.lang.Thread.run(Unknown Source) [?:1.8.0_74]
一切仍在处理中并正确关闭,我只是在我的日志中看到了这一点。不确定我的配置文件中是否存在冲突,这可能是问题的根源。我只是想确保这些更改不会导致问题,即使目前一切都按预期工作且没有错误。
最后是我的全部配置class:
@Configuration
@EnableJms
public class MQConfig {
private static final Logger logger = LogManager.getLogger(MQConfig.class.getName());
@Value("${mq.hostName}")
String host;
@Value("${mq.port}")
Integer port;
@Value("${mq.queueManager}")
String queueManager;
@Value("${mq.queueName}")
String queueName;
@Value("${mq.channel}")
String channel;
@Autowired
Environment environment;
@Bean
public DefaultMessageListenerContainer defaultMessageListenerContainer() {
DefaultMessageListenerContainer jmsListenerContainer = new DefaultMessageListenerContainer();
jmsListenerContainer.setConnectionFactory(mqConnectionFactory());
jmsListenerContainer.setDestinationResolver(destinationResolver());
jmsListenerContainer.setDestinationName(queueName);
jmsListenerContainer.setConcurrency("1");
jmsListenerContainer.setErrorHandler(errorHandler());
jmsListenerContainer.setSessionTransacted(true);
jmsListenerContainer.setSessionAcknowledgeMode(Session.CLIENT_ACKNOWLEDGE);
jmsListenerContainer.setAutoStartup(false);
return jmsListenerContainer;
}
@Bean
public MQConnectionFactory mqConnectionFactory() {
MQConnectionFactory mqConnectionFactory = new MQConnectionFactory();
try {
mqConnectionFactory.setHostName(host);
mqConnectionFactory.setPort(port);
mqConnectionFactory.setQueueManager(queueManager);
mqConnectionFactory.setTransportType(1);
} catch (JMSException ex) {
logger.error(ex.getStackTrace());
}
return mqConnectionFactory;
}
@Bean
public DynamicDestinationResolver destinationResolver() {
DynamicDestinationResolver destinationResolver = new DynamicDestinationResolver();
try {
Connection connection = mqConnectionFactory().createConnection();
Session session = connection.createSession(false, Session.AUTO_ACKNOWLEDGE);
destinationResolver.resolveDestinationName(session, queueName, false);
} catch (JMSException ex) {
logger.error(ex.getStackTrace());
}
return destinationResolver;
}
@Bean
public MQQueue mqQueue() {
MQQueue mqQueue = new MQQueue();
try {
mqQueue.setBaseQueueName(queueName);
mqQueue.setBaseQueueManagerName(queueManager);
} catch (JMSException ex) {
logger.error(ex.getStackTrace());
}
return mqQueue;
}
@Bean
public JmsErrorHandler errorHandler(){
return new JmsErrorHandler();
}
@Bean
public MQManager mqHoldManager(){
return new MQManager(host, port, queueName,
queueManager, channel);
}
@Bean
public MQInitializer mqInitializer(){
return new MQInitializer(environment);
}
}
想法?这么乱吗?
编辑:JMS 侦听器
@Component
public class MQConsumer {
@Resource(name = "mqHoldManager")
private MQManager mqHoldManager;
@Resource(name = "defaultMessageListenerContainer")
private DefaultMessageListenerContainer listenerContainer;
@Autowired
MQInitializer mqInitializer; /* To ensure this bean executes before Listener Setup not necessarily for any particular usage*/
final Logger logger = LogManager.getLogger(MQConsumer.class.getName());
private static ReportManager reportManager = new ReportManager();
private static boolean isFirstQueue = true;
@JmsListener(destination = "${mq.queueName}")
public void processOrder(String message) throws Exception {...}
我很困惑。
DefaultJmsListenerContainerFactory
用于带注释的 POJO 方法:
@JmsListener(...)
public void foo(String bar) {...}
工厂用于为方法创建侦听器容器。
使用您的替换配置,我没有看到您在 DefaultMessageListenerContainer
中配置消息侦听器。
通常你会有 container.setMessageListner(myListener)
,其中 myListener
是一个 MessageListener
,一个 SessionAwareMessageListener
或一个 MessageListenerAdapter
,它包装了一个 POJO 侦听器。
将 autoStartup
设置为 false 并且从不启动容器除了将容器 bean 添加到上下文之外什么都不做。
我不知道你怎么能用那个配置得到任何消息。
编辑
您正在使用 Spring 引导吗?如果是这样,它将为您创建一个默认 jmsListenerContainerFactory
- 这是我最好的猜测。
在这种情况下,您的停止代码并没有真正停止实际的容器 - 只是 "dummy" 一个从未启动过的容器。
我建议你给你的@JmsListener
一个id
,@Autowire
一个JmsListenerEndpointRegistry
然后打电话给registry.getListenerContainer("myListener").stop()
。
@JmsListener(id = "myListener", destination = "${mq.queueName}")
这是我之前 post 此处的一个单独但相关的问题:
我使用 spring 启动的 JMS 应用程序正确处理了所有内容,并且没有错误地关闭。为了让它工作,我从以下位置更改了一个 bean:
@Bean
public DefaultJmsListenerContainerFactory jmsListenerContainerFactory() {
DefaultJmsListenerContainerFactory factory = new DefaultJmsListenerContainerFactory();
factory.setConnectionFactory(mqConnectionFactory());
factory.setDestinationResolver(destinationResolver());
factory.setConcurrency("1");
factory.setErrorHandler(errorHandler());
factory.setSessionTransacted(true);
factory.setSessionAcknowledgeMode(Session.CLIENT_ACKNOWLEDGE);
return factory;
}
收件人:
@Bean
public DefaultMessageListenerContainer defaultMessageListenerContainer() {
DefaultMessageListenerContainer jmsListenerContainer = new DefaultMessageListenerContainer();
jmsListenerContainer.setConnectionFactory(mqConnectionFactory());
jmsListenerContainer.setDestinationResolver(destinationResolver());
jmsListenerContainer.setDestinationName(queueName);
jmsListenerContainer.setConcurrency("1");
jmsListenerContainer.setErrorHandler(errorHandler());
jmsListenerContainer.setSessionTransacted(true);
jmsListenerContainer.setSessionAcknowledgeMode(Session.CLIENT_ACKNOWLEDGE);
jmsListenerContainer.setAutoStartup(false);
return jmsListenerContainer;
}
问题是,我本可以只创建一个 "hotfix",因为我对 spring 的了解很少。当我偶然发现这个 post 时添加了更改的 bean jmsListenerContainer.setAutoStartup(false);
中的行: http://forum.spring.io/forum/spring-projects/integration/79176-illegalstateexception-no-message-listener-specified 因为没有 autoStartup
设置为 false 我在日志中的每条处理消息后得到这个:
java.lang.IllegalStateException: No message listener specified - see property 'messageListener'
at org.springframework.jms.listener.AbstractMessageListenerContainer.invokeListener(AbstractMessageListenerContainer.java:691) [spring-jms-4.2.4.RELEASE.jar:4.2.4.RELEASE]
at org.springframework.jms.listener.AbstractMessageListenerContainer.doExecuteListener(AbstractMessageListenerContainer.java:651) [spring-jms-4.2.4.RELEASE.jar:4.2.4.RELEASE]
at org.springframework.jms.listener.AbstractPollingMessageListenerContainer.doReceiveAndExecute(AbstractPollingMessageListenerContainer.java:315) [spring-jms-4.2.4.RELEASE.jar:4.2.4.RELEASE]
at org.springframework.jms.listener.AbstractPollingMessageListenerContainer.receiveAndExecute(AbstractPollingMessageListenerContainer.java:253) [spring-jms-4.2.4.RELEASE.jar:4.2.4.RELEASE]
at org.springframework.jms.listener.DefaultMessageListenerContainer$AsyncMessageListenerInvoker.invokeListener(DefaultMessageListenerContainer.java:1150) [spring-jms-4.2.4.RELEASE.jar:4.2.4.RELEASE]
at org.springframework.jms.listener.DefaultMessageListenerContainer$AsyncMessageListenerInvoker.executeOngoingLoop(DefaultMessageListenerContainer.java:1142) [spring-jms-4.2.4.RELEASE.jar:4.2.4.RELEASE]
at org.springframework.jms.listener.DefaultMessageListenerContainer$AsyncMessageListenerInvoker.run(DefaultMessageListenerContainer.java:1039) [spring-jms-4.2.4.RELEASE.jar:4.2.4.RELEASE]
at java.lang.Thread.run(Unknown Source) [?:1.8.0_74]
一切仍在处理中并正确关闭,我只是在我的日志中看到了这一点。不确定我的配置文件中是否存在冲突,这可能是问题的根源。我只是想确保这些更改不会导致问题,即使目前一切都按预期工作且没有错误。
最后是我的全部配置class:
@Configuration
@EnableJms
public class MQConfig {
private static final Logger logger = LogManager.getLogger(MQConfig.class.getName());
@Value("${mq.hostName}")
String host;
@Value("${mq.port}")
Integer port;
@Value("${mq.queueManager}")
String queueManager;
@Value("${mq.queueName}")
String queueName;
@Value("${mq.channel}")
String channel;
@Autowired
Environment environment;
@Bean
public DefaultMessageListenerContainer defaultMessageListenerContainer() {
DefaultMessageListenerContainer jmsListenerContainer = new DefaultMessageListenerContainer();
jmsListenerContainer.setConnectionFactory(mqConnectionFactory());
jmsListenerContainer.setDestinationResolver(destinationResolver());
jmsListenerContainer.setDestinationName(queueName);
jmsListenerContainer.setConcurrency("1");
jmsListenerContainer.setErrorHandler(errorHandler());
jmsListenerContainer.setSessionTransacted(true);
jmsListenerContainer.setSessionAcknowledgeMode(Session.CLIENT_ACKNOWLEDGE);
jmsListenerContainer.setAutoStartup(false);
return jmsListenerContainer;
}
@Bean
public MQConnectionFactory mqConnectionFactory() {
MQConnectionFactory mqConnectionFactory = new MQConnectionFactory();
try {
mqConnectionFactory.setHostName(host);
mqConnectionFactory.setPort(port);
mqConnectionFactory.setQueueManager(queueManager);
mqConnectionFactory.setTransportType(1);
} catch (JMSException ex) {
logger.error(ex.getStackTrace());
}
return mqConnectionFactory;
}
@Bean
public DynamicDestinationResolver destinationResolver() {
DynamicDestinationResolver destinationResolver = new DynamicDestinationResolver();
try {
Connection connection = mqConnectionFactory().createConnection();
Session session = connection.createSession(false, Session.AUTO_ACKNOWLEDGE);
destinationResolver.resolveDestinationName(session, queueName, false);
} catch (JMSException ex) {
logger.error(ex.getStackTrace());
}
return destinationResolver;
}
@Bean
public MQQueue mqQueue() {
MQQueue mqQueue = new MQQueue();
try {
mqQueue.setBaseQueueName(queueName);
mqQueue.setBaseQueueManagerName(queueManager);
} catch (JMSException ex) {
logger.error(ex.getStackTrace());
}
return mqQueue;
}
@Bean
public JmsErrorHandler errorHandler(){
return new JmsErrorHandler();
}
@Bean
public MQManager mqHoldManager(){
return new MQManager(host, port, queueName,
queueManager, channel);
}
@Bean
public MQInitializer mqInitializer(){
return new MQInitializer(environment);
}
}
想法?这么乱吗?
编辑:JMS 侦听器
@Component
public class MQConsumer {
@Resource(name = "mqHoldManager")
private MQManager mqHoldManager;
@Resource(name = "defaultMessageListenerContainer")
private DefaultMessageListenerContainer listenerContainer;
@Autowired
MQInitializer mqInitializer; /* To ensure this bean executes before Listener Setup not necessarily for any particular usage*/
final Logger logger = LogManager.getLogger(MQConsumer.class.getName());
private static ReportManager reportManager = new ReportManager();
private static boolean isFirstQueue = true;
@JmsListener(destination = "${mq.queueName}")
public void processOrder(String message) throws Exception {...}
我很困惑。
DefaultJmsListenerContainerFactory
用于带注释的 POJO 方法:
@JmsListener(...)
public void foo(String bar) {...}
工厂用于为方法创建侦听器容器。
使用您的替换配置,我没有看到您在 DefaultMessageListenerContainer
中配置消息侦听器。
通常你会有 container.setMessageListner(myListener)
,其中 myListener
是一个 MessageListener
,一个 SessionAwareMessageListener
或一个 MessageListenerAdapter
,它包装了一个 POJO 侦听器。
将 autoStartup
设置为 false 并且从不启动容器除了将容器 bean 添加到上下文之外什么都不做。
我不知道你怎么能用那个配置得到任何消息。
编辑
您正在使用 Spring 引导吗?如果是这样,它将为您创建一个默认 jmsListenerContainerFactory
- 这是我最好的猜测。
在这种情况下,您的停止代码并没有真正停止实际的容器 - 只是 "dummy" 一个从未启动过的容器。
我建议你给你的@JmsListener
一个id
,@Autowire
一个JmsListenerEndpointRegistry
然后打电话给registry.getListenerContainer("myListener").stop()
。
@JmsListener(id = "myListener", destination = "${mq.queueName}")