在没有遇到Permgen异常的情况下从Ant重新加载Tomcat应用程序的正确方法是什么?

时间:2009-07-28 20:58:22

标签: tomcat ant

我和我的团队正在开发一个Java Web应用程序,使用Tomcat作为我们的开发测试服务器,Ant作为我们的构建工具。我们的平台是Windows,但我们正在运行Cygwin中的Tomcat,因为我们首选的shell是bash。由于频繁重启应用程序以使新实现的功能可用于测试,我们遇到了内存耗尽的一致问题。

我们最初尝试重新加载应用程序,而无需使用提供的Catalina ant任务重新启动服务器。这适用于几次迭代但是我们总是耗尽Permgen空间。需要说明的是,这是指与Tomcat捆绑在一起的Ant任务,它启动,停止,部署,取消部署和列出应用程序。再次,这样做超过3次会导致Permgen内存错误,我们不得不手动杀死java进程。这就是为什么我们试图杀死服务器而不是应用程序。

我们的下一个想法是每次都重启服务器。我们一直存在的问题是服务器的关闭似乎并没有破坏java进程。最终,除非你手动监视并杀死它们,否则内存只会耗尽所有这些僵尸进程。我尝试过的实现如下:

  1. Executing the bootstrap jar directly

    <target name="tomcat.server.start" depends="tomcat.server.online" unless="tomcat.server.online">
        <java jar="${tomcat.home}/bin/bootstrap.jar" fork="true">
            <jvmarg value="-Dcatalina.home=${tomcat.home}" />
        </java>
        <waitfor maxwait="30" maxwaitunit="second" timeerty="tomcat.offline">
            <socket port="8081" server="127.0.0.1" />
        </waitfor>
        <fail if="tomcat.offline">Tomcat failed to come up.</fail>
    </target>
    
  2. Executing the shutdown|startup.bat files directly

    <target name="tomcat.server.start" depends="tomcat.server.online" unless="tomcat.server.online">
        <exec executable="bash" os="Windows" failonerror="true" >
            <arg line="-i ../tools/tomcat/bin/startup.bat />
        </exec>
        <waitfor maxwait="30" maxwaitunit="second" timeerty="tomcat.offline">
            <socket port="8081" server="127.0.0.1" />
        </waitfor>
        <fail if="tomcat.offline">Tomcat failed to come up.</fail>
    </target>
    
  3. 执行shutdown | startup.sh文件。

    <target name="tomcat.server.start" depends="tomcat.server.online" unless="tomcat.server.online">
        <exec executable="bash" failonerror="true" >
            <arg line="-i ../tools/tomcat/bin/startup.sh" />
        </exec>
        <waitfor maxwait="30" maxwaitunit="second" timeerty="tomcat.offline">
            <socket port="8081" server="127.0.0.1" />
        </waitfor>
        <fail if="tomcat.offline">Tomcat failed to come up.</fail>
    </target>
    
  4. 所有上述方法都使java进程保持活动状态,至少在我实现它们的时候。

    此时我不知道该怎么做。我听说过在你没有重装容器或应用程序的情况下进行爆炸式开发,但我从未发现如何做到这一点。我不知道这里接受的最佳做法是什么。

    我的问题主要是在进行测试时运行Permgen空间。如果我首先提出错误的问题,那么请随意更正标题。我对任何建议都持开放态度。

    谢谢!

3 个答案:

答案 0 :(得分:4)

使用Tomcat Ant Tasks。 Catalina-ant.jar包含在大多数Tomcat发行版中。通常,将此与try-catch Ant任务相结合是很好的。

<taskdef name="start" classname="org.apache.catalina.ant.StartTask" />
<taskdef name="stop" classname="org.apache.catalina.ant.StopTask" />
<taskdef name="try" classname="ise.antelope.tasks.TryTask"/>
<try>
    <stop url="${url}"
        username="${username}"
        password="${password}"
        path="${path}"/>
    <catch>
        <echo>${path} stop failed.</echo>
    </catch>
</try>
 <try>
    <start url="${url}"
        username="${username}"
        password="${password}"
        path="${path}"/>
    <catch>
        <echo>${path} start failed.</echo>
    </catch>
</try>

如果这是一个纯内存问题,你可能想要在JVM启动时调整一些内存条目(我经常这样做,但从来没有像thisthis那样清楚地写出来。 ):

-XX:+UseConcMarkSweepGC
-XX:+CMSPermGenSweepingEnabled
-XX:+CMSClassUnloadingEnabled
-XX:MaxPermSize=256m

答案 1 :(得分:1)

有完成这些事情的Tomcat Ant任务:

http://raibledesigns.com/wiki/Wiki.jsp?page=TomcatAntTasks

也许他们可以提供帮助。

答案 2 :(得分:0)

我建议将Tomcat作为服务运行。然后,您可以使用SSHEXEC任务。

从我们的build.xml中编辑:

<target name="_restartService">
    <sshexec
        host="${deploy.server.host}"
        username="${deploy.username}"
        password="${deploy.password}"
        timeout="120000"
        command="restartContainer.bat ${deploy.client}"
        trust="true" />
</target>

restartContainer.bat文件调用NET STOP%serviceName%,清理日志,临时文件等,然后调用NET START%serviceName%。

它对我们在开发,舞台演出和制作方面非常有用。您只需要安装SSH服务器。