本文介绍了几个调整JSP和servlet的一些非常实用的方法它可使你的servlet和JSP页面响应更快扩展性更强而且在用户数增加的情况下系统负载会呈现出平滑上长的趋势我将通过一些实际例子和配置方法使得你的应用程序的性能有出人意料的提升其中某些调优技术是在你的编程工作中实现的而另一些技术是与应用服务器的配置相关的我们将详细地描述怎样通过调整servlet和JSP页面来提高你的应用程序的总体性能在阅读本文之前你还需要有基本的servlet和JSP的知识
方法一在servlet的init()方法中缓存数据
当应用服务器初始化servlet实例之后为客户端请求提供服务之前它会调用这个servlet的init()方法在一个servlet的生命周期中init()方法只会被调用一次通过在init()方法中缓存一些静态的数据或完成一些只需要执行一次的耗时的操作就可大大地提高系统性能
例如通过在init()方法中建立一个JDBC连接池是一个最佳例子假设我们是用jdbc的DataSource接口来取得数据库连接在通常的情况下我们需要通过JNDI来取得具体的数据源我们可以想象在一个具体的应用中如果每次SQL请求都要执行一次JNDI查询的话那系统性能将会急剧下降解决方法是如下代码它通过缓存DataSource使得下一次SQL调用时仍然可以继续利用它
publicclassControllerServlet
extendsHttpServlet
{
privatejavaxsqlDataSource
testDS=null;
publicvoidinit(ServletConfigconfig)
throwsServletException
{
superinit(config);
Contextctx=null;
try
{
ctx=newInitialContext();
testDS=(javaxsqlDataSource)
ctxlookup("jdbc/testDS");
}
catch(NamingExceptionne)
{
neprintStackTrace();
}
catch(Exceptione)
{
eprintStackTrace();
}
}
publicjavaxsqlDataSourcegetTestDS()
{
returntestDS;
}
}
方法:禁止servlet和JSP自动重载(autoreloading)
Servlet/JSP提供了一个实用的技术即自动重载技术它为开发人员提供了一个好的开发环境当你改变servlet和JSP页面后而不必重启应用服务器然而这种技术在产品运行阶段对系统的资源是一个极大的损耗因为它会给JSP引擎的类装载器(classloader)带来极大的负担因此关闭自动重载功能对系统性能的提升是一个极大的帮助
方法:不要滥用HttpSession
在很多应用中我们的程序需要保持客户端的状态以便页面之间可以相互联系但不幸的是由于HTTP具有天生无状态性从而无法保存客户端的状态因此一般的应用服务器都提供了session来保存客户的状态
在JSP应用服务器中是通过HttpSession对像来实现session的功能的但在方便的同时它也给系统带来了不小的负担因为每当你获得或更新session时系统者要对它进行费时的序列化操作你可以通过对HttpSession的以下几种处理方式来提升系统的性能
如果没有必要就应该关闭JSP页面中对HttpSession的缺省设置如果你没有明确指定的话每个JSP页面都会缺省地创建一个HttpSession如果你的JSP中不需要使用session的话那可以通过如下的JSP页面指示符来禁止它
<%@pagesession="false"%>
不要在HttpSession中存放大的数据对像如果你在HttpSession中存放大的数据对像的话每当对它进行读写时应用服务器都将对其进行序列化从而增加了系统的额外负担你在HttpSession中存放的数据对像越大那系统的性能就下降得越快
当你不需要HttpSession时尽快地释放它当你不再需要session时你可以通过调用HttpSessioninvalidate()方法来释放它
尽量将session的超时时间设得短一点在JSP应用服务器中有一个缺省的session的超时时间当客户在这个时间之后没有进行任何操作的话系统会将相关的session自动从内存中释放超时时间设得越大系统的性能就会越低因此最好的方法就是尽量使得它的值保持在一个较低的水平
方法:将页面输出进行压缩
压缩是解决数据冗余的一个好的方法特别是在网络带宽不够发达的今天有的浏览器支持gzip(GNUzip)进行来对HTML文件进行压缩这种方法可以戏剧性地减少HTML文件的下载时间因此如果你将servlet或JSP页面生成的HTML页面进行压缩的话那用户就会觉得页面浏览速度会非常快但不幸的是不是所有的浏览器都支持gzip压缩但你可以通过在你的程序中检查客户的浏览器是否支持它下面就是关于这种方法实现的一个代码片段
publicvoiddoGet(HttpServletRequestrequest
HttpServletResponseresponse)
throwsIOException
ServletException
{
OutputStreamout=null
Stringencoding=
requestgetHeader("AcceptEncoding");
if(encoding!=
null&&encodingindexOf("gzip")!=)
{
requestsetHeader
("ContentEncoding""gzip");
out=newGZIPOutputStream
(requestgetOutputStream());
}
elseif(encoding!=
null&&encodingindexOf("compress")!=)
{
requestsetHeader
("ContentEncoding""compress");
out=newZIPOutputStream
(requestgetOutputStream());
}
else
{
out=requestgetOutputStream();
}
}
方法:使用线程池
应用服务器缺省地为每个不同的客户端请求创建一个线程进行处理并为它们分派service()方法当service()方法调用完成后与之相应的线程也随之撤消由于创建和撤消线程会耗费一定的系统资源这种缺省模式降低了系统的性能但所幸的是我们可以通过创建一个线程池来改变这种状况
另外我们还要为这个线程池设置一个最小线程数和一个最大线程数在应用服务器启动时它会创建数量等于最小线程数的一个线程池当客户有请求时相应地从池从取出一个线程来进行处理当处理完成后再将线程重新放入到池中
如果池中的线程不够地话系统会自动地增加池中线程的数量但总量不能超过最大线程数通过使用线程池当客户端请求急剧增加时系统的负载就会呈现的平滑的上升曲线从而提高的系统的可伸缩性
方法:选择正确的页面包含机制
在JSP中有两种方法可以用来包含另一个页面
使用include指示符(<%@includeefile=”testjsp”%>)
使用jsp指示符(<jsp:includeepage=”testjsp”flush=”true”/>)
在实际中我发现如果使用第一种方法的话可以使得系统性能更高
方法:正确地确定javabean的生命周期
JSP的一个强大的地方就是对javabean的支持通过在JSP页面中使用<jsp:useBean>标签可以将javabean直接插入到一个JSP页面中它的使用方法如下
<jsp:useBeanid="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之争)决定了你的应用程序的性能重要是你要对这种平台有一个较为深入的了解这样你才能从根本上对自己的应用程序做一个优化!