许多Web应用、企业应用涉及到长时间的操作,例如复杂的数据库查询或繁重的XML处理等,虽然这些任务主要由数据库系统或中间件完成,但任务执行的结果仍旧要借助JSP才能发送给用户。本文介绍了一种通过改进前端表现层来改善用户感觉、减轻服务器负载的办法。
当JSP调用一个必须长时间运行的操作,且该操作的结果不能(在服务器端)缓冲,用户每次请求该页面时都必须长时间等待。很多时候,用户会失去耐心,接着尝试点击浏览器的刷新按钮,最终失望地离开。
本文介绍的技术是把繁重的计算任务分离开来,由一个独立的线程运行,从而解决上述问题。当用户调用JSP页面时,JSP页面会立即返回,并提示用户任务已经启动且正在执行;JSP页面自动刷新自己,报告在独立线程中运行的繁重计算任务的当前进度,直至任务完成。
一、模拟任务
首先我们设计一个TaskBean类,它实现java.lang.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);
}
<