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