`
king_tt
  • 浏览: 2287893 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
社区版块
存档分类
最新评论

详细解析JSP编程中进度条的设计实例

 
阅读更多
详细解析JSP编程中进度条的设计实例(1)

许多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调用:

<ccid_nobr></ccid_nobr>
<ccid_code>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; } }</ccid_code>


TaskBean 包含的“繁重任务”是计算1+2+3…+100的值,不过它不通过100*(100+1)/2=5050公式计算,而是由run()方法调用work() 方法100次完成计算。work()方法的代码如下所示,其中调用Thread.sleep()是为了确保任务总耗时约10秒。

<ccid_nobr></ccid_nobr>
<ccid_code>protected void work(){ try{ Thread.sleep(sleep); counter++; sum += counter; } catch (InterruptedException e){ setRunning(false); } }</ccid_code>


status.jsp页面通过调用下面的getPercent()方法获得任务的完成状况:

<ccid_nobr></ccid_nobr>
<ccid_code>public synchronized int getPercent(){ return counter; }</ccid_code>


如果任务已经启动,isStarted()方法将返回true:

<ccid_nobr></ccid_nobr>
<ccid_code>public synchronized boolean isStarted(){ return started; }</ccid_code>


如果任务已经完成,isCompleted()方法将返回true:

<ccid_nobr></ccid_nobr>
<ccid_code>public synchronized boolean isCompleted(){ return counter == 100; }</ccid_code>


如果任务正在运行,isRunning()方法将返回true:

<ccid_nobr></ccid_nobr>
<ccid_code>public synchronized boolean isRunning(){ return running; }</ccid_code>


SetRunning()方法由start.jsp或stop.jsp调用,当running参数是true时。SetRunning()方法还要将任务标记为“已经启动”。调用setRunning(false)表示要求run()方法停止执行。

<ccid_nobr></ccid_nobr>
<ccid_code>public synchronized void setRunning(boolean running){ this.running = running; if (running) started = true; }</ccid_code>


任务执行完毕后,调用getResult()方法返回计算结果;如果任务尚未执行完毕,它返回null:

<ccid_nobr></ccid_nobr>
<ccid_code>public synchronized Object getResult(){ if (isCompleted()) return new Integer(sum); else return null; }</ccid_code>


当running 标记为true、completed标记为false时,run()方法调用work()。在实际应用中,run()方法也许要执行复杂的SQL查询、解 析大型XML文档,或者调用消耗大量CPU时间的EJB方法。注意“繁重的任务”可能要在远程服务器上执行。报告结果的JSP页面有两种选择:或者等待任 务结束,或者使用一个进度条。

<ccid_nobr></ccid_nobr>
<ccid_code>public void run() { try { setRunning(true); while (isRunning() &amp;&amp; !isCompleted()) work(); } finally { setRunning(false); } }</ccid_code>


二、启动任务

start.jsp是web.xml部署描述符中声明的欢迎页面,web.xml的内容是:

<ccid_nobr></ccid_nobr>
<ccid_code>&lt;?xml version="1.0" encoding="GB2312"?&gt; &lt;!DOCTYPE web-app PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN" "http://java.sun.com/dtd/web-app_2_3.dtd"&gt; &lt;web-app&gt; &lt;welcome-file-list&gt; &lt;welcome-file&gt;start.jsp&lt;/welcome-file&gt; &lt;/welcome-file-list&gt; &lt;/web-app&gt;</ccid_code>


start.jsp启动一个专用的线程来运行“繁重的任务”,然后把HTTP请求传递给status.jsp。

start.jsp页面利用<usebean>标记创建一个TaskBean的实例,将scope属性定义为session使得对于来自同一浏览器的HTTP请求,其他页面也能提取到同一个Bean对象。start.jsp通过调用session.removeAttribute("task")确保<usebean>创建了一个新的Bean对象,而不是提取一个旧对象(例如,同一个用户会话中更早的JSP页面所创建的Bean对象)。 <br><br>下面是start.jsp页面的代码清单: <br><br></usebean></usebean> <ccid_nobr></ccid_nobr>
<ccid_code>&lt;% session.removeAttribute("task"); %&gt; &lt;jsp:useBean id="task" scope="session" class="test.barBean.TaskBean"/&gt; &lt;% task.setRunning(true); %&gt; &lt;% new Thread(task).start(); %&gt; &lt;jsp:forward page="status.jsp"/&gt;</ccid_code>


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页面的多个线程。
详细解析JSP编程中进度条的设计实例 (2)

三、任务进度

status.jsp页面利用一个HTML进度条向用户显示任务的执行情况。首先,status.jsp利用<usebean>标记获得start.jsp页面创建的Bean对象:<br><br><br><br><br><br><br><br><br><br></usebean> <ccid_nobr></ccid_nobr>
<ccid_code>&lt;jsp:useBean id="task" scope="session" class="test.barBean.TaskBean"/&gt;</ccid_code>


为了及时反映任务执行进度,status.jsp会自动刷新。JavaScript代码setTimeout("location=′status.jsp′", 1000)将每隔1000毫秒刷新页面,重新请求status.jsp,不需要用户干预。

<ccid_nobr></ccid_nobr>
<ccid_code>&lt;HTML&gt; &lt;HEAD&gt; &lt;TITLE&gt;JSP进度条&lt;/TITLE&gt; &lt;% if (task.isRunning()) { %&gt; &lt;SCRIPT LANGUAGE="JavaScript"&gt; setTimeout("location=′status.jsp′", 1000); &lt;/SCRIPT&gt; &lt;% } %&gt; &lt;/HEAD&gt; &lt;BODY&gt;</ccid_code>


进度条实际上是一个HTML表格,包含10个单元,即每个单元代表任务总体的10%进度。

<ccid_nobr></ccid_nobr>
<ccid_code>&lt;H1 ALIGN="CENTER"&gt;JSP进度条&lt;/H1&gt; &lt;H2 ALIGN="CENTER"&gt;</ccid_code>


结果:

<ccid_nobr></ccid_nobr>
<ccid_code>&lt;%= task.getResult() %&gt;&lt;BR&gt; &lt;% int percent = task.getPercent();%&gt; &lt;%= percent %&gt;% &lt;/H2&gt; &lt;TABLE WIDTH="60%" ALIGN="CENTER" BORDER=1 CELLPADDING=0 CELLSPACING=2&gt; &lt;TR&gt; &lt;% for (int i = 10; i &lt;= percent; i += 10){ %&gt; &lt;TD WIDTH="10%" BGCOLOR="#000080"&gt; &lt;/TD&gt; &lt;% } %&gt; &lt;% for (int i = 100; i &gt; percent; i -= 10){ %&gt; &lt;TD WIDTH="10%"&gt; &lt;/TD&gt; &lt;%} %&gt; &lt;/TR&gt; &lt;/TABLE&gt;</ccid_code>


任务执行情况分下面几种状态:正在执行,已完成,尚未开始,已停止:

<ccid_nobr></ccid_nobr>
<ccid_code>&lt;TABLE WIDTH="100%" BORDER=0 CELLPADDING=0 CELLSPACING=0&gt; &lt;TR&gt; &lt;TD ALIGN="CENTER"&gt; &lt;% if (task.isRunning()) { %&gt;</ccid_code>


正在执行

<ccid_nobr></ccid_nobr>
<ccid_code>&lt;% } else { %&gt; &lt;% if (task.isCompleted()) { %&gt;</ccid_code>


完成

<ccid_nobr></ccid_nobr>
<ccid_code>&lt;% } else if (!task.isStarted()){ %&gt;</ccid_code>


尚未开始

<ccid_nobr></ccid_nobr>
<ccid_code>&lt;% } else { %&gt;</ccid_code>


已停止

<ccid_nobr></ccid_nobr>
<ccid_code>&lt;% } %&gt; &lt;% } %&gt; &lt;/TD&gt; &lt;/TR&gt;</ccid_code>


页面底部提供了一个按钮,用户可以用它来停止或重新启动任务:

<ccid_nobr></ccid_nobr>
<ccid_code>&lt;TR&gt; &lt;TD ALIGN="CENTER"&gt; &lt;BR&gt; &lt;% if (task.isRunning()) { %&gt; &lt;FORM METHOD="GET" ACTION="stop.jsp"&gt; &lt;INPUT TYPE="SUBMIT" VALUE="停止"&gt; &lt;/FORM&gt; &lt;% } else { %&gt; &lt;FORM METHOD="GET" ACTION="start.jsp"&gt; &lt;INPUT TYPE="SUBMIT" VALUE="开始"&gt; &lt;/FORM&gt; &lt;% } %&gt; &lt;/TD&gt; &lt;/TR&gt; &lt;/TABLE&gt; &lt;/BODY&gt;&lt;/HTML&gt;</ccid_code>


只要不停止任务,约10秒后浏览器将显示出计算结果5050:

四、停止任务

stop.jsp页面把running标记设置成false,从而停止当前的计算任务:

<ccid_nobr></ccid_nobr>
<ccid_code>&lt;jsp:useBean id="task" scope="session" class="test.barBean.TaskBean"/&gt; &lt;% task.setRunning(false); %&gt; &lt;jsp:forward page="status.jsp"/&gt;</ccid_code>


注意最早的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构造进度条是相当容易的,真正困难的是如何将它用到实际应用之中,特别是获得后台任务的进度信息,但这个问题没有通用的答案,每一种后台执行的任务都有它自己的特点,必须按照具体情况具体分析。

[自]:http://www.iyuanma.com/info/25/24057_20059268384.htm
分享到:
评论

相关推荐

    JSP编程进度条设计实例

    ### JSP编程进度条设计实例解析 在现代Web开发中,动态网页的实时反馈功能尤为重要,其中进度条作为用户交互的重要组成部分,能够显著提升用户体验。本文将深入解析一个经典的JSP编程进度条设计实例,旨在展示如何...

    java,jsp,ajax进度条(jsp)

    综上所述,"java,jsp,ajax进度条(jsp)"项目是一个结合了后端Java编程、服务器端渲染以及客户端异步通信的实例。它展示了如何通过这些技术协作,提供一种流畅的用户界面,实时反馈大文件操作的进度,提高了用户对长...

    jsp中使用ajax实现web进度条

    "jsp中使用ajax实现web进度条"这个主题就是关于如何在Java Server Pages (JSP)中利用Asynchronous JavaScript and XML (AJAX)技术来创建一个动态显示文件上传进度的组件。下面将详细介绍这一过程中的关键知识点。 1...

    文件上传组件JspSmartUpload

    - **处理上传请求**:在Servlet的`doPost`方法中,实例化`SmartUpload`对象,调用其方法解析上传请求,获取上传的文件信息。 - **保存文件**:根据业务需求,将接收到的文件保存到服务器的指定位置。 - **反馈结果**...

    jsp页面内实现批量上传下载.rar

    在这个实例中,"SmartUpload20071101(完美版)"很可能包含了SmartUpload的源码和使用说明。SmartUpload通过解析HTTP请求中的文件数据,将每个文件保存到服务器的指定位置,并返回一个上传成功的响应。 下载部分,...

    上传下载]仿163网盘无刷新文件上传 for Jsp_fileupload_jsp.rar

    【标题解析】 "上传下载]仿163网盘无刷新文件上传 ...这个项目提供了实现无刷新文件上传的一个实例,对于学习JSP和AJAX的开发者来说,是一个有价值的参考资料,可以通过分析源代码了解如何在实际项目中集成这些技术。

    jsp文件上传fgsfgasfgs

    综上所述,JSP文件上传涉及到后端的数据解析、文件处理、安全防护,以及前端的表单设计等多个方面。开发者在实现这一功能时,必须充分考虑性能、安全性和用户体验。对于提供的文件名“jsp文件上传”,很可能是用于...

    jsp版的多文件上传源代码

    总结来说,这个“jsp版的多文件上传源代码”是一个实践性的学习资源,涵盖了Java Web开发中文件上传的各个环节,包括前端交互、HTTP协议理解、Servlet编程以及安全性考虑等,对于提升开发者在这一领域的技能十分有...

    Java 批量上传文件实例

    本实例将详细讲解如何在Java中实现这一功能,结合使用jar包和JSP来构建一个完整的解决方案。 首先,我们要了解批量上传的基本原理。批量上传通常涉及到多个文件的选取和发送,这需要前端页面(如JSP)提供一个多选...

    Android应用源码之安卓图片上传和文件上传带jsp服务端源码.zip

    - 服务端,这里使用了JSP(JavaServer Pages)技术,JSP的`&lt;form&gt;`标签可以接收上传的文件,`Servlet`负责处理请求,解析文件并保存到服务器。 2. **文件上传** - 文件上传的流程与图片上传类似,但可能需要处理...

    Java写的一个mp3播放器.rar

    下面将详细讨论这个项目中涉及的关键知识点。 1. **Java基础**: - **类与对象**:MP3播放器的核心是通过类来组织代码,如`MediaPlayer`、`Playlist`等,它们代表不同的实体和功能。 - **异常处理**:Java的异常...

    java各种文件上传例子

    2. 在Servlet或JSP中实例化SmartUpload对象。 3. 初始化SmartUpload,设置上传路径、最大文件大小等参数。 4. 使用SmartUpload的上传方法处理请求,它可以自动解析请求中的文件。 5. 遍历上传的文件,进行保存或操作...

    java上传下载.

    在Java编程领域,上传和下载是常见的功能,尤其是在Web应用中。Java JSP(Java Server Pages)技术常常被用于构建动态网页,实现用户界面与服务器端的交互,包括文件的上传和下载。以下是对这些知识点的详细说明: ...

    java开源包1

    开发它是用于在UTF-8 Oracle实例中使用ASCII编码的Oracle 数据库中来正确的传输非ASCII字符。 Java模板语言 Beetl Beetl,是Bee Template Language的缩写,它绝不是简单的另外一种模板引擎,而是新一代的模板引擎,...

    java开源包2

    开发它是用于在UTF-8 Oracle实例中使用ASCII编码的Oracle 数据库中来正确的传输非ASCII字符。 Java模板语言 Beetl Beetl,是Bee Template Language的缩写,它绝不是简单的另外一种模板引擎,而是新一代的模板引擎,...

    java开源包3

    开发它是用于在UTF-8 Oracle实例中使用ASCII编码的Oracle 数据库中来正确的传输非ASCII字符。 Java模板语言 Beetl Beetl,是Bee Template Language的缩写,它绝不是简单的另外一种模板引擎,而是新一代的模板引擎,...

    java开源包6

    开发它是用于在UTF-8 Oracle实例中使用ASCII编码的Oracle 数据库中来正确的传输非ASCII字符。 Java模板语言 Beetl Beetl,是Bee Template Language的缩写,它绝不是简单的另外一种模板引擎,而是新一代的模板引擎,...

    java开源包5

    开发它是用于在UTF-8 Oracle实例中使用ASCII编码的Oracle 数据库中来正确的传输非ASCII字符。 Java模板语言 Beetl Beetl,是Bee Template Language的缩写,它绝不是简单的另外一种模板引擎,而是新一代的模板引擎,...

    java开源包10

    开发它是用于在UTF-8 Oracle实例中使用ASCII编码的Oracle 数据库中来正确的传输非ASCII字符。 Java模板语言 Beetl Beetl,是Bee Template Language的缩写,它绝不是简单的另外一种模板引擎,而是新一代的模板引擎,...

    java开源包8

    开发它是用于在UTF-8 Oracle实例中使用ASCII编码的Oracle 数据库中来正确的传输非ASCII字符。 Java模板语言 Beetl Beetl,是Bee Template Language的缩写,它绝不是简单的另外一种模板引擎,而是新一代的模板引擎,...

Global site tag (gtag.js) - Google Analytics