一、模拟任务
首先我们设计一个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()方法停止执行。
public synchronized void setRunning(boolean running) {
this.running = running;
if (running)
started = true;
}
任务执行完毕后,调用getResult()方法返回计算结果;如果任务尚未执行完毕,它返回null:
public synchronized Object getResult() {
if (isCompleted())
return new Integer(sum);
else
return null;
}
当running标记为true、completed标记为false时,run()方法调用work()。在实际应用中,run()方法也许要执行复杂的SQL查询、解析大型XML文档,或者调用消耗大量CPU时间的EJB方法。注意“繁重的任务”可能要在远程服务器上执行。报告结果的JSP页面有两种选择:或者等待任务结束,或者使用一个进度条。
public void run() {
try {
setRunning(true);
while (isRunning() && !isCompleted())
work();
} finally {
setRunning(false);
}
}
二、启动任务
start.jsp是web.xml部署描述符中声明的欢迎页面,web.xml的内容是:
<?xml version="1.0" encoding="GB2312"?>
<!DOCTYPE web-app
PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
"http://java.sun.com/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:useBean id="task" scope="session"
class="test.barBean.TaskBean"/>
<% task.setRunning(true); %>
<% new Thread(task).start(); %>
<jsp:forward page="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线程中运行。
start.jsp在创建线程之前就把TaskBean的running标记设置成了true,这样,即使当JSP线程已开始执行status.jsp而任务线程的run()方法尚未启动,也能够确保用户会得到“任务已开始运行”的状态报告。
将running标记设置成true、启动任务线程这两行代码可以移入TaskBean构成一个新的方法,然后由JSP页面调用这个新方法。一般而言,JSP页面应当尽量少用Java代码,即我们应当尽可能地把Java代码放入Java类。不过本例中我们不遵从这一规则,把new Thread(task).start()直接放入start.jsp突出表明JSP线程创建并启动了任务线程。
在JSP页面中操作多线程必须谨慎,注意JSP线程和其它线程实际上是并发执行的,就象在桌面应用程序中,我们用一个线程来处理GUI事件,另外再用一个或多个线程来处理后台任务。不过在JSP环境中,考虑到多个用户同时请求某一个页面的情况,同一个JSP页面可能会在多个线程中同时运行;另外,有时同一个用户可能会向同一个页面发出多个请求,虽然这些请求来自同一个用户,它们也会导致服务器同时运行一个JSP页面的多个线程。
三、任务进度
status.jsp页面利用一个HTML进度条向用户显示任务的执行情况。首先,status.jsp利用<jsp:useBean>标记获得start.jsp页面创建的Bean对象:
<jsp:useBean id="task" scope="session"
class="test.barBean.TaskBean"/>
为了及时反映任务执行进度,status.jsp会自动刷新。JavaScript代码setTimeout("location='status.jsp'", 1000)将每隔1000毫秒刷新页面,重新请求status.jsp,不需要用户干预。
<HTML>
<HEAD>
<TITLE>JSP进度条</TITLE>
<% if (task.isRunning()) { %>
<SCRIPT LANGUAGE="JavaScript">
setTimeout("location='status.jsp'", 1000);
</SCRIPT>
<% } %>
</HEAD>
<ODY>
进度条实际上是一个HTML表格,包含10个单元——即每个单元代表任务总体的10%进度。
<H1 ALIGN="CENTER">JSP进度条</H1>
<H2 ALIGN="CENTER">
结果: <%= task.getResult(<) %><BR>
% int percent = task.getPercent(); %>
<%= percent %>%
</H2>
<TABLE WIDTH="60%" ALIGN="CENTER"
BORDER=1 CELLPADDING=0 CELLSPACING=2>
<TR>
<% for (int i = 10; i <= percent; i += 10) { %>
<TD WIDTH="10%" BGCOLOR="#000080"> </TD>
<% } %>
<% for (int i = 100; i > percent; i -= 10) { %>
<TD WIDTH="10%"> </TD>
<% } %>
</TR>
</TABLE>
任务执行情况分下面几种状态:正在执行,已完成,尚未开始,已停止:
<TABLE WIDTH="100%" BORDER=0 CELLPADDING=0 CELLSPACING=0>
<TR>
<TD ALIGN="CENTER">
<% if (task.isRunning()) { %>
正在执行
<% } else { %>
<% if (task.isCompleted()) { %>
完成
<% } else if (!task.isStarted()) { %>
尚未开始
<% } else { %>
已停止
<% } %>
<% } %>
</TD>
</TR>
页面底部提供了一个按钮,用户可以用它来停止或重新启动任务:
<TR>
<TD ALIGN="CENTER">
<BR>
<% if (task.isRunning()) { %>
<FORM METHOD="GET" ACTION="stop.jsp">
<INPUT TYPE="SUBMIT" VALUE="停止">
</FORM>
<% } else { %>
<FORM METHOD="GET" ACTION="start.jsp">
<INPUT TYPE="SUBMIT" VALUE="开始">
</FORM>
<% } %>
</TD>
</TR>
</TABLE>
</BODY></HTML>
只要不停止任务,约10秒后浏览器将显示出计算结果5050:
四、停止任务
stop.jsp页面把running标记设置成false,从而停止当前的计算任务:
<jsp:useBean id="task" scope="session"
class="test.barBean.TaskBean"/>
<% task.setRunning(false); %>
<jsp:forward page="status.jsp"/>
注意最早的Java版本提供了Thread.stop方法,但JDK从1.2版开始已经不赞成使用Thread.stop方法,所以我们不能直接调用Thread.stop()。
第一次运行本文程序的时候,你会看到任务的启动有点延迟;同样地,第一次点击“停止”按钮时也可以看到任务并没有立即停止运行(特别是如果机器配置较低的话,延迟的感觉更加明显),这些延迟都是由于编译JSP页面导致的。编译好JSP页面之后,应答速度就要快多了。
五、实际应用
进度条不仅使得用户界面更加友好,而且对服务器的性能也有好处,因为进度条会不断地告诉用户当前的执行进度,用户不会再频繁地停止并重新启动(刷新)当前的任务。另一方面,创建单独的线程来执行后台任务也会消耗不少资源,必要时可考虑通过一个线程池来实现Thread对象的重用。另外,频繁地刷新进度页面也增加了网络通信开销,所以务必保持进度页面简洁短小。
在实际应用中,后台执行的繁重任务可能不允许停止,或者它不能提供详细的执行进度数据。例如,查找或更新关系数据库时,SQL命令执行期间不允许中途停止——不过如果用户表示他想要停止或中止任务,程序可以在SQL命令执行完毕后回退事务。
解析XML文档的时候,我们没有办法获知已解析内容精确的百分比。如果用DOM解析XML文档,直到解析完成后才得到整个文档树;如果用SAX,虽然可以知道当前解析的内容,但通常不能确定还有多少内容需要解析。在这些场合,任务的执行进度只能靠估计得到。
估计一个任务需要多少执行时间通常是很困难的,因为它涉及到许多因素,即使用实际测试的办法也无法得到可靠的结论,因为服务器的负载随时都在变化之中。一种简单的办法是测量任务每次执行所需时间,然后根据最后几次执行的平均时间估算。如果要提高估计时间的精确度,应当考虑实现一种针对应用特点的算法,综合考虑多种因素,例如要执行的SQL语句类型、要解析的XML模式的复杂程度,等等。
结束语:本文例子显示出用JSP、Java、HTML和JavaScript构造进度条是相当容易的,真正困难的是如何将它用到实际应用之中,特别是获得后台任务的进度信息,但这个问题没有通用的答案,每一种后台执行的任务都有它自己的特点,必须按照具体情况具体分析。
分享到:
相关推荐
在本场景中,我们讨论的是如何使用Java Server Pages (JSP) 技术来实现一个登录过程中的进度条效果。这通常是通过JavaScript、CSS以及后台的Java逻辑协同工作来完成的。 首先,我们需要理解JSP的基础。JSP是一种...
本项目"JSP进度条使用线程制作"正是为了解决这个问题,它展示了如何通过多线程技术在JSP中实现一个动态更新的进度条。 首先,我们需要理解线程的概念。在Java中,线程是程序执行的最小单元,每个线程都有自己的程序...
通过这个简单的JSP进度条例子,你可以学习到如何结合前端和后端技术实现动态交互的Web应用。在实际项目中,还可以进一步优化,比如使用WebSockets实现实时双向通信,提供更好的用户体验。同时,对于大型项目,理解...
在这个"jsp上传实现进度条"的主题中,我们将深入探讨如何在JSP中实现这一功能。 首先,我们需要了解文件上传的基本流程。用户在前端通过HTML表单选择文件后,这些文件会被发送到服务器。在传统的HTTP请求中,一旦...
本示例是关于如何在JSP(JavaServer Pages)环境下实现一个简单的进度条。虽然这个示例没有提供百分比进度的功能,但我们可以从中学习到基础的JSP编程概念以及如何创建动态网页效果。 首先,我们需要理解JSP的基本...
【标题】: "用jsp实现的进度条" 这个主题涉及到的是在Web开发中如何使用JavaServer Pages (JSP) 技术来创建一个实时更新的进度条展示功能。在Web应用程序中,进度条通常用于告知用户某个长时间运行的任务(如文件...
"jsp进度条文件上传"这个主题聚焦于如何在JavaServer Pages (JSP)环境中,利用Ajax技术来实现文件上传时的进度条展示,以提供更好的用户体验。这通常涉及到前端与后端的协同工作,以及JavaScript库如Prototype.js的...
在实际应用中,JSP进度条常用于各种场景,如文件上传、数据库查询、大计算任务等。例如,在文件上传过程中,可以根据已上传文件的大小与总大小的比例来更新进度条,让用户了解文件上传的实时状态,提高用户的耐心和...
### JSP 页面进度条实现详解 #### 一、引言 在Web开发中,进度条是一种常见的用户界面元素,用于直观地展示任务的完成情况。本文将深入探讨一个基于JSP(JavaServer Pages)的简单进度条实现方案,通过分析提供的...
根据提供的文件信息,我们可以深入探讨JSP中的八种进度条实现方法。首先,我们通过分析提供的部分代码示例来理解其工作原理和技术要点。 ### 第一种进度条:动态加载进度条 #### 技术背景与原理 在第一种进度条的...
要实现进度条的显示,就要知道两个参数,上传的大小和总文件的大小 重写的解析器提供了一个上传过程事件,在上传过程中不断触发,然后用已上传的大 小/总大小,计算上传的百分比,然后将此数据缓存起来,前端发送异步...
至于提供的两个HTML文件(1.html、2.html),可能包含了一些具体的进度条实现示例,你可以打开查看它们是如何将上述概念应用到实际项目中的。记住,每个项目的需求可能会有所不同,所以要根据实际情况调整代码。 ...
本项目"java,jsp,ajax进度条(jsp)"结合了这三种技术,旨在实现一个功能强大的进度条,用于监控大文件的上传或下载过程,提升用户体验。 首先,让我们深入理解这三个技术的基础概念: 1. **Java**:Java是一种跨...
本示例将介绍两种不同的JSP网页进度条实现方式,帮助开发者了解如何创建此类功能。 ### 1. 基于HTML5 `<progress>` 元素 HTML5 提供了一个内置的 `<progress>` 元素,可以用来显示任务的进度。在JSP页面中,你可以...
本示例关注的是如何在使用JSP(JavaServer Pages)时实现文件上传并显示进度条,这对于用户交互体验至关重要,特别是处理大文件时。下面我们将详细探讨这个主题。 首先,`upfile.jsp`是一个关键文件,它包含了处理...
**JSP+Ajax实现进度条** 1. **服务器端**:在`ProgressBarJsp.jsp`中,你可以定义一个Servlet方法来处理进度更新。例如,你可以创建一个持久化任务,该任务在后台运行并定期更新其进度。JSP页面可以通过Ajax请求这...
"JSP文件上传 支持进度条显示"这个项目是针对JSP环境设计的一个AJAX Web上传程序,它允许用户上传大文件(如单个文件100M),并具有文件上传进度条的可视化功能。以下将详细解析这个项目的知识点: 1. **JSP(Java...
【JSP编程进度条设计实例】是一个关于提升用户体验和优化服务器负载的技术实践。在很多Web应用和企业应用中,执行复杂数据库查询或大量XML处理等长时间任务时,尽管主要工作由数据库系统或中间件承担,但最终结果仍...
1、支持多个文件上传, 2、进度条样式可以自定义, 3、按照百分制的进度显示
"jsp中使用ajax实现web进度条"这个主题就是关于如何在Java Server Pages (JSP)中利用Asynchronous JavaScript and XML (AJAX)技术来创建一个动态显示文件上传进度的组件。下面将详细介绍这一过程中的关键知识点。 1...