0

我正在编写一个在 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 感到困惑的地方,如果您能提供任何帮助和建议,我将不胜感激。

4

1 回答 1

1

我强烈建议您使用控制 Tomcat 实例生命周期的包装器来包装 Tomcat 实例。这样的包装器就是Java Service Wrapper。较旧的版本(我相信是 3.2.3)是“免费的”,并且适用于较新的 Tomcat 实例。

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

于 2012-07-13T08:23:53.293 回答