java - 如何从 Java 干净地启动和停止 Tomcat,为什么我的方法不能重复工作?

标签 java tomcat quartz-scheduler

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

  1. 确保 Tomcat 已停止(在 Tomcat 下运行的 Solr 可防止索引目录被删除或复制)
  2. 必要时删除旧索引目录
  3. 启动Tomcat
  4. 确保 Tomcat 和 Solr 应用程序正在运行
  5. 运行索引作业
  6. 停止 Tomcat
  7. 确保 Tomcat 已停止
  8. 将索引目录复制到存档

我决定让启动和停止 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 感到困惑的地方,如果您能提供任何帮助和建议,我将不胜感激。

最佳答案

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

然后您的控制应用程序与包装器“对话”以启动/停止 Tomcat 应用程序。这种方法有很多好处。其中之一是您不会受到 Tomcat 应用程序挂起和您正在测试的端口不再回复的影响。

关于java - 如何从 Java 干净地启动和停止 Tomcat,为什么我的方法不能重复工作?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11464463/

相关文章:

java - 我们怎么知道Heap要设置多大呢?

java - WebLogic 上 Jenkins 的 LinkageError

java - 非法访问 : this web application instance has been stopped already. 无法加载 org.apache.log4j.spi.NOPLoggerRepository

java - 无法访问 servlet

java - 在 TomCat 服务器上使用 JavaServlets 的 Cronjob

java - 如何从 JTable 中获取选中的 RadioButton

hibernate - Tomcat 不停止

java - 如何在所有线程完成之前保持 JVM Activity ?

java - 使用 Quartz 或任何其他 java api 进行动态作业调度

java - 调用父方法并仍然保留当前变量