许多Web应用、企业应用涉及到长时间的操作,例如复杂的数据库查询或繁重的XML处理等,虽然这些任务主要由数据库系统或中间件完成,但任务执行的结果仍旧要借助JSP才能发送给用户。本文介绍了一种通过改进前端表现层来改善用户感觉、减轻服务器负载的办法。
当JSP调用一个必须长时间运行的操作,且该操作的结果不能(在服务器端)缓冲,用户每次请求该页面时都必须长时间等待。很多时候,用户会失去耐心,接着尝试点击浏览器的刷新按钮,最终失望地离开。
本文介绍的技术是把繁重的计算任务分离开来,由一个独立的线程运行,从而解决上述问题。当用户调用JSP页面时,JSP页面会立即返回,并提示用户任务已经启动且正在执行;JSP页面自动刷新自己,报告在独立线程中运行的繁重计算任务的当前进度,直至任务完成。
一、模拟任务
首先我们设计一个TaskBean类,它实现ng.Runnable接口,其run()方法在一个由JSP页面(start.jsp)启 动的独立线程中运行。终止run()方法执行由另一个JSP页面stop.jsp负责。TaskBean类还实现了 java.io.Serializable接口,这样JSP页面就可以将它作为JavaBean调用:
packagetest.barBean;
importjava.io.Serializable;
publicclassTaskBeanimplementsRunnable,Serializable{
privateintcounter;
privateintsum;
privatebooleanstarted;
privatebooleanrunning;
privateintsleep;
publicTaskBean(){
counter=0;
sum=0;
started=false;
running=false;
sleep=100;
}
}
TaskBean包含的"繁重任务"是计算 1+2+3…+100的值,不过它不通过100*(100+1)/2=5050公式计算,而是由run()方法调用work()方法100次完成计算。 work()方法的代码如下所示,其中调用Thread.sleep()是为了确保任务总耗时约10秒。
protectedvoidwork(){
try{
Thread.sleep(sleep);
counter++;
sum+=counter;
}catch(InterruptedExceptione){
setRunning(false);
}
}
status.jsp页面通过调用下面的getPercent()方法获得任务的完成状况:
publicsynchronizedintgetPercent(){
returncounter;
}
如果任务已经启动,isStarted()方法将返回true:
publicsynchronizedbooleanisStarted(){
returnstarted;
}
如果任务已经完成,isCompleted()方法将返回true:
publicsynchronizedbooleanisCompleted(){
returncounter==100;
}
如果任务正在运行,isRunning()方法将返回true:
publicsynchronizedbooleanisRunning(){
returnrunning;
}
SetRunning()方法由start.jsp或stop.jsp调用,当running参数是true时。SetRunning()方法还要将任务标记为"已经启动"。调用setRunning(false)表示要求run()方法停止执行。
publicsynchronizedvoidsetRunning(booleanrunning){
this.running=running;
if(running)
started=true;
}
任务执行完毕后,调用getResult()方法返回计算结果;如果任务尚未执行完毕,它返回null:
publicsynchronizedObjectgetResult(){
if(isCompleted())
returnnewInteger(sum);
else
returnnull;
}
当running标记为true、completed标记为false时,run()方法调用work()。在实际应用中,run()方法也许要执行复 杂的SQL查询、解析大型XML文档,或者调用消耗大量CPU时间的EJB方法。注意"繁重的任务"可能要在远程服务器上执行。报告结果的JSP页面有两 种选择:或者等待任务结束,或者使用一个进度条。
publicvoidrun(){
try{
setRunning(true);
while(isRunning()&&!isCompleted())
work();
}finally{
setRunning(false);
}
}
二、启动任务
start.jsp是web.xml部署描述符中声明的欢迎页面,web.xml的内容是:
<?xmlversion="1.0"encoding="GB2312"?>
<!DOCTYPEweb-app
PUBLIC"-//SunMicrosystems,Inc.//DTDWebApplication2.3//EN"
"/dtd/web-app_2_3.dtd">
<web-app>
<welcome-file-list>
<welcome-file>start.jsp</welcome-file>
</welcome-file-list>
</web-app>
start.jsp启>动一个专用的线程来运行"繁重的任务",然后把HTTP请求传递给status.jsp。
start.jsp页面利用<jsp:useBean>标记创建一个TaskBean的实例,将scope属性定义为session使得对于来自 同一浏览器的HTTP请求,其他页面也能提取到同一个Bean对象。start.jsp通过调用 session.removeAttribute("task")确保<jsp:useBean>创建了一个新的Bean对象,而不是提取一个旧对 象(例如,同一个用户会话中更早的JSP页面所创建的Bean对象)。
下面是start.jsp页面的代码清单:
<%session.removeAttribute("task");%>
<jsp:useBeanid="task"scope="session"
class="test.barBean.TaskBean"/>
<%task.setRunning(true);%>
<%newThread(task).start();%>
<jsp:forwardpage="status.jsp"/>
start.jsp创建并设置好TaskBean对象之后,接着创建一个Thread,并将Bean对象作为一个Runnable实例传入。调用start()方法时新创建的线程将执行TaskBean对象的run()方法。
现在有两个线程在并发执行:执行JSP页面的线程(称之为"JSP线程"),由JSP页面创建的线程(称之为"任务线程")。接下 来,start.jsp利用调用status.jsp,status.jsp显示出进度条以及任务的执行情况。注意status.jsp和 start.jsp在同一个JSP线程中运行。