方法一在servlet的init()方法中缓存数据
当应用服务器初始化servlet实例之后为客户端请求提供服务之前他会调用这个servlet的init()方法在一个servlet的生命周期中init()方法只会被调用一次通过在init()方法中缓存一些静态的数据或完成一些只需要执行一次的耗时的操作就可大大地提高系统性能
例如通过在init()方法中建立一个JDBC连接池是个最好例子假设我们是用jdbc的DataSource接口来取得数据库连接在通常的情况下我们需要通过JNDI来取得具体的数据源我们能够想象在一个具体的应用中假如每次SQL请求都要执行一次JNDI查询的话那系统性能将会急剧下降解决方法是如下代码他通过缓存DataSource使得下一次SQL调用时仍然能够继续利用他
public class ControllerServlet extends HttpServlet{
private javaxsqlDataSource testDS = null;
public void init(ServletConfig config) throws ServletException{
superinit(config);
Context ctx = null;
try{
ctx = new InitialContext();
testDS = (javaxsqlDataSource)ctxlookup("jdbc/testDS");
}catch(NamingException ne){neprintStackTrace();}
}catch(Exception e){eprintStackTrace();}
}
public javaxsqlDataSource getTestDS(){
return testDS;
}
}
方法 :禁止servlet和JSP 自动重载(autoreloading)
Servlet/JSP提供了一个实用的技术即自动重载技术他为研发人员提供了一个好的研发环境当您改变servlet和JSP页面后而不必重启应用服务器然而这种技术在产品运行阶段对系统的资源是个极大的损耗因为他会给JSP引擎的类装载器(classloader)带来极大的负担因此关闭自动重载功能对系统性能的提升是个极大的帮助
方法 : 不要滥用HttpSession
在很多应用中我们的程式需要保持客户端的状态以便页面之间能够相互联系但不幸的是由于HTTP具备天生无状态性从而无法保存客户端的状态因此一般的应用服务器都提供了session来保存客户的状态在JSP应用服务器中是通过HttpSession对像来实现session的功能的但在方便的同时他也给系统带来了不小的负担因为每当您获得或更新session时系统者要对他进行费时的序列化操作您能够通过对HttpSession的以下几种处理方式来提升系统的性能
假如没有必要就应该关闭JSP页面中对HttpSession的缺省配置 假如您没有明确指定的话每个JSP页面都会缺省地创建一个HttpSession假如您的JSP中无需使用session的话那能够通过如下的JSP页面指示符来禁止他
<%@ page session="false"%>
不要在HttpSession中存放大的数据对像假如您在HttpSession中存放大的数据对像的话每当对他进行读写时应用服务器都将对其进行序列化从而增加了系统的额外负担您在HttpSession中存放的数据对像越大那系统的性能就下降得越快
当您无需HttpSession时尽快地释放他当您不再需要session时您能够通过调用HttpSessioninvalidate()方法来释放他尽量将session的超时时间设得短一点在JSP应用服务器中有一个缺省的session的超时时间当客户在这个时间之后没有进行任何操作的话系统会将相关的session自动从内存中释放超时时间设得越大系统的性能就会越低因此最好的方法就是尽量使得他的值保持在一个较低的水平
方法 : 将页面输出进行压缩
压缩是解决数据冗余的一个好的方法特别是在网络带宽不够发达的今天有的浏览器支持gzip(GNU zip)进行来对HTML文档进行压缩这种方法能够戏剧性地减少HTML文档的下载时间因此假如您将servlet或JSP页面生成的HTML页面进行压缩的话那用户就会觉得页面浏览速度会很快但不幸的是不是任何的浏览器都支持gzip压缩但您能够通过在您的程式中检查客户的浏览器是否支持他下面就是关于这种方法实现的一个代码片段
public void doGet(HttpServletRequest request HttpServletResponse response)
throws IOException ServletException {
OutputStream out = null;
String encoding = requestgetHeader("AcceptEncoding");
if (encoding != null && encodingindexOf("gzip") != ){
requestsetHeader("ContentEncoding" "gzip");
out = new GZIPOutputStream(requestgetOutputStream());
}
else if (encoding != null && encodingindexOf("comdivss") != ){
requestsetHeader("ContentEncoding" "comdivss");
out = new ZIPOutputStream(requestgetOutputStream());
}else{
out = requestgetOutputStream();
}
}
方法 : 使用线程池
应用服务器缺省地为每个不同的客户端请求创建一个线程进行处理并为他们分派service()方法当service()方法调用完成后和之相应的线程也随之撤消由于创建和撤消线程会耗费一定的系统资源这种缺省模式降低了系统的性能但所幸的是我们能够通过创建一个线程池来改变这种状况
另外我们还要为这个线程池配置一个最小线程数和一个最大线程数在应用服务器启动时他会创建数量等于最小线程数的一个线程池当客户有请求时相应地从池从取出一个线程来进行处理当处理完成后再将线程重新放入到池中假如池中的线程不够地话系统会自动地增加池中线程的数量但总量不能超过最大线程数通过使用线程池当客户端请求急剧增加时系统的负载就会呈现的平滑的上升曲线从而提高的系统的可伸缩性
方法 : 选择正确的页面包含机制
在JSP中有两种方法能够用来包含另一个页面
使用include指示符
<%@ includee file=”testjsp” %>
使用jsp指示符
<jsp:includee page=”testjsp” flush=”true”/>
在实际中发现假如使用第一种方法的话能够使得系统性能更高
方法 :正确地确定javabean的生命周期
JSP的一个强大的地方就是对javabean的支持通过在JSP页面中使用jsp:useBean标签能够将javabean直接插入到一个JSP页面中他的使用方法如下
<jsp:useBean id="name" scope="page|request|session|application"
class="packageclassName" type="typeName">
</jsp:useBean>
其中scope属性指出了这个bean的生命周期缺省的生命周期为page假如您没有正确地选择bean的生命周期的话他将影响系统的性能
举例来说假如您只想在一次请求中使用某个bean但您却将这个bean的生命周期配置成了session那当这次请求结束后这个bean将仍然保留在内存中除非session超时或用户关闭浏览器这样会耗费一定的内存并无谓的增加了JVM垃圾收集器的工作量因此为bean配置正确的生命周期并在bean的使命结束后尽快地清理他们会使用系统性能有一个提高
其他一些有用的方法
在字符串连接操作中尽量不使用“+”操作符在java编程中我们常常使用“+”操作符来将几个字符串连接起来但您或许从来没有想到过他居然会对系统性能造成影响吧?由于字符串是常量因此JVM会产生一些临时的对像您使用的“+”越多生成的临时对像就越多这样也会给系统性能带来一些影响解决的方法是用StringBuffer对像来代替“+”操作符
避免使用Systemoutprintln()方法由于Systemoutprintln()是一种同步调用即在调用他时磁盘I/O操作必须等待他的完成因此我们要尽量避免对他的调用但我们在调试程式时他又是个必不可少的方便工具为了解决这个矛盾我建议您最好使用Logj工具他既能够方便调试而不会产生Systemoutprintln()这样的方法
ServletOutputStream 和 PrintWriter的权衡使用PrintWriter可能会带来一些小的开销因为他将任何的原始输出都转换为字符流来输出因此假如使用他来作为页面输出的话系统要负担一个转换过程而使用ServletOutputStream作为页面输出的话就不存在一个问题但他是以二进制进行输出的因此在实际应用中要权衡两者的利弊
总结
本文的目的是通过对servlet和JSP的一些调优技术来极大地提高您的应用程式的性能并因此提升整个JEE应用的性能通过这些调优技术您能够发现其实并不是某种技术平台(比如JEE和NET之争)决定了您的应用程式的性能重要是您要对这种平台有一个较为深入的了解这样您才能从根本上对自己的应用程式做一个优化