当JSP调用一个必须长时间运行的操作,且该操作的结果不能(在服务器端)缓冲,用户每次请求该页面时都必须长时间等待。很多时候,用户会失去耐心,接着尝试点击浏览器的刷新按钮,最终失望地离开。
本文介绍的技术是把繁重的计算任务分离开来,由一个独立的线程运行,从而解决上述问题。当用户调用JSP页面时,JSP页面会立即返回,并提示用户任务已经启动且正在执行;JSP页面自动刷新自己,报告在独立线程中运行的繁重计算任务的当前进度,直至任务完成。
一、模拟任务
首先我们设计一个TaskBean类,它实现java.lang.Runnable接口,其run()方法在一个由JSP页面(start.jsp)启动的独立线程中运行。终止run()方法执行由另一个JSP页面stop.jsp负责。TaskBean类还实现了java.io.Serializable接口,这样JSP页面就可以将它作为JavaBean调用:
package test.barBean; import java.io.Serializable; public class TaskBean implements Runnable, Serializable { private int counter; private int sum; private boolean started; private boolean running; private int sleep; public TaskBean() { 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秒。
protected void work() { try { Thread.sleep(sleep); counter++; sum += counter; } catch (InterruptedException e) { setRunning(false); } }
|
status.jsp页面通过调用下面的getPercent()方法获得任务的完成状况:
public synchronized int getPercent() { return counter; }
|
如果任务已经启动,isStarted()方法将返回true:
public synchronized boolean isStarted() { return started; }
|
如果任务已经完成,isCompleted()方法将返回true:
public synchronized boolean isCompleted() { return counter == 100; }
|
如果任务正在运行,isRunning()方法将返回true:
public synchronized boolean isRunning() { return running; }
|
SetRunning()方法由start.jsp或stop.jsp调用,当running参数是true时。SetRunning()方法还要将任务标记为“已经启动”。调用setRunning(false)表示要求run()方法停止执行。
【编辑推荐】
- JavaOne 2009第三天:微软与Sun/Oracle携手并进
- 开发高可移植性J2ME的软件
- Java虚拟机(JVM)中的内存设置详解
- Sun拟推Java软件商店 克隆苹果成功经验
- 四个有害的Java编码习惯