我工作的地方同时使用 Apache Tomcat 6.0 和 Jetty 6。我们主要使用 Jetty 进行测试(它非常适合在 JUnit 测试中运行嵌入式)和 Tomcat 进行生产。
默认情况下,Tomcat 会在用户请求时即时编译 JSP。但这会导致第一次命中的性能下降。它还突出了 bizarre bugs在 Tomcat 的 JSP 编译器中。
Tomcat documentation给出了在构建时使用 Ant 预编译 JSP 的建议(并且 Maven 插件也可用)......但是生成的 WAR 包含 Tomcat 特定的东西,例如PageContextImpl.proprietaryEvaluate,所以我们不能在 Jetty 中使用它。
是否有一些标志或设置我们可以在某处使用来强制 Tomcat 在 WAR 初始化后立即预编译所有 JSP?为此,我们准备在启动时等待更长的时间。
提前:我知道有一种方法可以通过在 web.xml 中为一个 JSP 显式标识一个/servlet/load-on-startup 标记来精确地预编译 一个 JSP。但是对于数十个甚至数百个 JSP,这将变得难以管理。
最佳答案
http://www.devshed.com/c/a/BrainDump/Tomcat-Capacity-Planning/
project name="pre-compile-jsps" default="compile-jsp-servlets">
<!-- Private properties. -- >
<property name="webapp.dir" value="${basedir}/webapp-dir"/>
<property name="tomcat.home" value="/opt/tomcat"/>
<property name="jspc.pkg.prefix" value="com.mycompany"/>
<property name="jspc.dir.prefix" value="com/mycompany"/>
<!-- Compilation properties. -->
<property name="debug" value="on"/>
<property name="debuglevel" value="lines,vars,source"/>
<property name="deprecation" value="on"/>
<property name="encoding" value="ISO-8859-1"/>
<property name="optimize" value="off"/>
<property name="build.compiler" value="modern"/>
<property name="source.version" value="1.5"/>
<!-- Initialize Paths. -->
<path id="jspc.classpath">
<fileset dir="${tomcat.home}/bin">
<include name="*.jar"/>
</fileset>
<fileset dir="${tomcat.home}/server/lib">
<include name="*.jar"/>
</fileset>
<fileset dir="${tomcat.home}/common/i18n">
<include name="*.jar"/>
</fileset>
<fileset dir="${tomcat.home}/common/lib">
<include name="*.jar"/>
</fileset>
<fileset dir="${webapp.dir}/WEB-INF">
<include name="lib/*.jar"/>
</fileset>
<pathelement location="${webapp.dir}/WEB-INF/classes"/>
<pathelement location="${ant.home}/lib/ant.jar"/>
<pathelement location="${java.home}/../lib/tools.jar"/>
</path>
<property name="jspc.classpath" refid="jspc.classpath"/>
<!--========================================================== -->
<!-- Generates Java source and a web.xml file from JSP files. -->
<!-- ==========================================================
-->
<target name="generate-jsp-java-src">
<mkdir dir="${webapp.dir}/WEB-INF/jspc-src/${jspc.dir.prefix}"/>
<taskdef classname="org.apache.jasper.JspC" name="jasper2">
<classpath>
<path refid="jspc.classpath"/>
</classpath>
</taskdef>
<touch file="${webapp.dir}/WEB-INF/jspc-web.xml"/>
<jasper2 uriroot="${webapp.dir}"
package="${jspc.pkg.prefix}"
webXmlFragment="${webapp.dir}/WEB-INF/jspc-web.xml"
outputDir="${webapp.dir}/WEB-INF/jspc-src/${jspc.dir.prefix}"
verbose="1"/>
</target>
<!-- ========================================================== -->
<!-- Compiles (generates Java class files from) the JSP servlet -->
<!-- source code that was generated by the JspC task. -->
<!-- ========================================================== -->
<target name="compile-jsp-servlets" depends="generate-jsp-java-src">
<mkdir dir="${webapp.dir}/WEB-INF/classes"/>
<javac srcdir="${webapp.dir}/WEB-INF/jspc-src"
destdir="${webapp.dir}/WEB-INF/classes"
includes="**/*.java"
debug="${debug}"
debuglevel="${debuglevel}"
deprecation="${deprecation}"
encoding="${encoding}"
optimize="${optimize}"
source="${source.version}">
<classpath>
<path refid="jspc.classpath"/>
</classpath>
</javac>
</target>
<!-- ========================================================= -->
<!-- Cleans any pre-compiled JSP source, classes, jspc-web.xml -->
<!-- ========================================================= -->
<target name="clean">
<delete dir="${webapp.dir}/WEB-INF/jspc-src"/>
<delete dir="${webapp.dir}/WEB-INF/classes/${jspc.dir.prefix}"/>
<delete file="${webapp.dir}/WEB-INF/jspc-web.xml"/>
</target>
</project
此构建文件将找到您的 webapp 的所有 JSP 文件,将它们编译成 servlet 类,并为这些 JSP servlet 类生成 servlet 映射。它生成的 servlet 映射 ping 必须进入您的 webapp 的 WEB-INF/web.xml 文件,但是很难编写一个知道如何以可重复的方式将 servlet 映射插入到您的 web.xml 文件中的 Ant 构建文件构建文件运行的时间。相反,我们使用了一个 XML 实体包含,这样每次构建文件运行时生成的 servlet 映射都会进入一个新文件,并且可以通过 XML 实体包含机制将该 servlet 映射文件插入到您的 web.xml 文件中。要使用它,你的 webapp 的 WEB-INF/web.xml 必须在文件顶部有一个特殊的实体声明,加上对 web.xml 文件内容中你希望 servlet 映射文件所在的实体的引用包括。以下是一个空的 servlet 2.5 webapp 的 web.xml 文件经过这些修改后的样子:
<!DOCTYPE jspc-webxml [
<!ENTITY jspc-webxml SYSTEM "jspc-web.xml">
]>
<web-app xmlns=http://java.sun.com/xml/ns/javaee
xmlns:xsi=http://www.w3.org/2001/ XMLSchema-instance
xsi:schemaLocation="http:// java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/
javaee/web-app_2_5.xsd"
version="2.5">
<!-- We include the JspC-generated mappings here. -->
&jspc-webxml;
<!-- Non-generated web.xml content goes here. -->
</web-app>
确保您的 webapp 的 web.xml 文件在文件顶部一直包含内联 DTD(DOCTYPE 标记),并且在其下方包含 servlet 2.5 web-app 架构声明。然后,将生成的 servlet 映射插入到 web.xml 文件中的任何位置,将实体引用 &jspc-webxml; .请记住,实体引用以与号 ( & ) 开头,然后是实体的名称,最后以分号 ( ; ) 结尾。
要使用构建文件,只需编辑它并将顶部的所有属性设置为与您的设置匹配的值,然后像这样运行它:
$ ant -f pre-compile-jsps.xml
关于java - 如何让 Tomcat 在启动时预编译 JSP?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/497830/