我正在开发一个将部署到Tomcat的Web应用程序。启动Tomcat后,我使用servlet(在web.xml中)调用Java类:
<web-app> <display-name>Consumer</display-name> <servlet> <servlet-name>start</servlet-name> <servlet-class>com.test.sample.Consumer</servlet-class> <load-on-startup>1</load-on-startup> </servlet> </web-app>
我的Consumer.java订阅了AMQP服务器上的队列。我通过使用while (true)循环来实现此目的,该循环在独立的Java程序中可以正常工作。Itt也可以在Web应用程序的上下文中使用,但是我永远无法停止Tomcat服务器(在NetBeans IDE中),并且我相信while循环是罪魁祸首。这是一些代码:
while (true)
public class Consumer { public Consumer() consume(); } private void consume() ... while (true) { // Await incoming messages from queue // Process message } } }
有没有更好的方法来解决这个问题?还是要发出信号才能退出循环?
谢谢!
更新为使用ServletContextListener:
public final class ApplicationListener implements ServletContextListener { private ScheduledExecutorService scheduler; public ApplicationListener() { } @Override public void contextDestroyed(ServletContextEvent event) { System.out.println("***** Stopping Consumer *****"); scheduler.shutdownNow(); } @Override public void contextInitialized(ServletContextEvent event) { System.out.println("***** Starting Consumer *****"); scheduler = Executors.newSingleThreadScheduledExecutor(); scheduler.scheduleAtFixedRate(new ScheduledConsumer(), 0, 15000, TimeUnit.MILLISECONDS); } public class ScheduledConsumer implements Runnable { @Override public void run() { Consumer k = new Consumer(); k.consumeOnce(); } } }
我有一些建议,但是它们要求您稍微修改一下体系结构,以便更好地使用容器环境。
Servlet容器支持可获取各种事件通知的“侦听器”。具体来说,其中之一是ServletContextListener当上下文(通过Web contextInitialized方法)投入使用(通过方法)和何时将其退出服务(通过contextDestroyed方法)时得到通知的通知。
ServletContextListener
contextInitialized
contextDestroyed
我的建议是执行以下操作:
Consumer
consume()
consumeOnce
Thread
volatile boolean stop
Consumer.consumeOnce
Thread.sleep
true
Thread.interrupt
我确定我缺少一些确切的细节,但这是一般的想法。当Tomcat关闭时,您的代码将收到关闭通知,您可以干净地终止自己的循环线程。您可能需要提供一种方法,使之Consumer在尝试获取Thread.interrupt信号时不中止,从而中止尝试消耗其所消耗的任何东西(例如,停止等待从空队列中拉出对象)。(例如,如果您使用一个Object.wait()命令来等待监视器通知,那么您将需要更改它,以便它使用带有超时的等待命令,这样您就不会永远阻塞。)
Object.wait()