如何从Java干净地启动和停止Tomcat以及为什么我的方式不能重复工作?

时间:2012-07-13 05:10:11

标签: java tomcat quartz-scheduler

我正在编写一个在Windows上运行的Quartz应用程序,并调用Lucene和Solr来运行索引作业。它运行一系列作业,每个作业都包含以下步骤:

  1. 确保Tomcat已停止(在Tomcat下运行的Solr会阻止删除或复制索引目录)
  2. 必要时删除旧索引目录
  3. 启动Tomcat
  4. 确保Tomcat和Solr应用正在运行
  5. 运行索引作业
  6. 停止Tomcat
  7. 确保Tomcat已停止
  8. 将索引目录复制到存档
  9. 我决定使用启动和停止Tomcat的代码设置在Startup.bat,Shutdown.bat和Catalina.bat中设置的系统属性,然后使用“start”和“stop”参数调用Bootstrap.main。这适用于一次迭代,但是当我尝试使用Quartz运行时,我设置了两次迭代。

    当我的代码在第一次迭代结束时关闭Tomcat时,显示了所有常用消息,包括

    INFO: Stopping ProtocolHandler ["http-bio-5918"]
    

    (我正在使用端口5918)但是当它试图在第二次迭代开始时启动Tomcat时,它得到了错误:

    SEVERE: Failed to initialize end point associated with ProtocolHandler ["http-bio-5918"]
    java.net.BindException: Address already in use: JVM_Bind <null>:5918
    

    SEVERE: Failed to initialize connector [Connector[HTTP/1.1-5918]]
    org.apache.catalina.LifecycleException: Failed to initialize component [Connector[HTTP/1.1-5918]]
    

    我在命令提示符窗口中运行netstat -an,它确认端口5918正在使用中。我用来检查Tomcat是否正在运行的代码没有什么特别之处。我在互联网上的各个地方见过。

    public boolean isTomcatRunning(String url) {
      boolean isRunning = false;
    
      try {
        new URL(url).openConnection().connect();
        isRunning = true;
      } catch (IOException e) {
        isRunning = false;
      }
      return isRunning;
    }
    

    但它显然告诉我Tomcat没有运行。

    正如我所说,我通过调用Bootstrap.main(new String [] {“start”})和Bootstrap.main(new String [] {“stop”})来启动和停止Tomcat。唯一特别的是,当我简单地调用Bootstrap.main(new String [] {“start”})时,它似乎没有返回(我还没有等待足够的时间来查看它是否是悬挂或只是花了很长时间),所以我一直在线程中运行它。

    也许这会导致问题,因为看起来Catalina.bat没有做任何特别的事情,它从启动回来就好了。我想知道是否需要进行额外的设置才能使它在主线程中运行启动而不会挂起。

    无论如何,这是我在Quartz应用程序中启动和停止Tomcat时的困惑,我将非常感谢您提供的任何帮助和建议。

1 个答案:

答案 0 :(得分:1)

我强烈建议您使用包装器来包装Tomcat实例,该包装器控制Tomcat实例的生命周期。这样的包装器是Java Service Wrapper。较旧的版本(我认为是3.2.3)是“免费的”,并且可以与较新的Tomcat实例一起使用。

然后,您的控制应用程序将“与”包装器“对话”以启动/停止Tomcat应用程序。这种方法有很多好处。其中之一就是您不再需要挂起Tomcat应用程序,而且您正在测试的端口不再响应。