小学生也可以在因特网上发布 HTML 网页。但是,小学生的网页和专业开发的网站有质的区别。网页设计人员(或者 HTML 开发人员)必须理解颜色、用户、生产流程、网页布局、浏览器兼容性、图像创建和 JavaScript 等等。设计漂亮的网站需要做大量的工作,大多数 Java 开发人员更注重创建优美的对象接口,而不是用户界面。JavaServer Pages (JSP) 技术为网页设计人员和 Java 开发人员提供了一种联系钮带。
如果您开发过大型 Web 应用程序,您就理解 变化 这个词的含义。“模型-视图-控制器”(MVC) 就是用来帮助您控制变化的一种设计模式。MVC 减弱了业务逻辑接口和数据接口之间的耦合。Struts 是一种 MVC 实现,它将 Servlet 2.2 和 JSP 1.1 标记(属于 J2EE 规范)用作实现的一部分。尽管您可能永远不会用 Struts 实现一个系统,但了解一下 Struts 或许使您能将其中的一些思想用于您以后的 Servlet 的 JSP 实现中。
在本文中,我将以一个 JSP 文件为起点讨论该网页的优缺点,该文件中使用的元素可能是您所熟悉的。随后我将讨论 Struts,并说明它是如何控制您的 Web 项目中的变化并提高专业化水平的。最后,我将重新开发这个简单的 JSP 文件,在开发过程中我已顾及到网页设计人员和变化。
|
|
JavaServer Page (JSP) 文件只是审视 servlet 的另一种方式。JSP 文件的概念使我们能够将 Java servlet 看作一个 HTML 网页。JSP 消除了 Java 代码中经常出现的讨厌的 print()
语句。JSP 文件首先被预处理为 .java
文件,然后再编译为 .class
文件。如果您使用的是 Tomcat,则可以在 work
目录下查看预处理后的 .java
文件。别的容器可能将 .java
和 .class
文件存储在其他位置;这个位置与容器有关。图 1 说明了从 JSP 文件到 servlet 的流程。
(这与 Microsoft 的 Active Server Page (ASP) 明显不同。ASP 被编译到内存中,而不是编译到一个单独的文件中。)
在小型 JSP 应用程序中,经常会看到数据、业务逻辑和用户界面被组合在一个代码模块中。此外,应用程序通常还包含用来控制应用程序流程的逻辑。清单 1 和图 2 展示了允许用户加入一个邮件列表的一个简单 JSP 文件。
清单 1. join.jsp -- 一个简单的请求和响应 JSP 文件
<%@ page language="java" %> <%@ page import="business.util.Validation" %> <%@ page import="business.db.MailingList" %> <% String error = ""; String email = request.getParameter("email"); // 是否有电子邮件地址 if( email!=null ) { // 验证输入... if( business.util.Validation.isValidEmail(email) ) { // 存储输入... try { business.db.MailingList.AddEmail(email); } catch (Exception e) { error = "Error adding email address to system. " + e; } if( error.length()==0 ) { %> // 重定向到欢迎页... <jsp:forward page="welcome.html"/> <% } } else { // 设置错误消息并重新显示网页 error = email + " is not a valid email address, please try again."; } } else { email = ""; } %> <html> <head> <title>Join Mailing List</title> </head> <body> <font color=red><%=error%></font><br> <h3>Enter your email to join the group</h3> <form action="join.jsp" name="joinForm"> <input name="email" id="email" value=<%=email%>></input> <input type=submit value="submit"> </form> </body> </html> |
图 2. 在简单的请求和响应中,JSP 文件设置数据、控制到下一个网页的流程并创建 HTML
这个邮件列表 JSP 文件是一个独立的、自主完成所有任务的模块。未包含在这个 JSP 文件中的仅有代码是包含在 isValidEmail()
中的实际验证代码和将电子邮件地址存入数据库的代码。(将 isValidEmail()
方法分离到可重用的代码中似乎是当然的选择,但我曾见过直接嵌入网页中的 isValidEmail()
代码。单页方法的优点是易于理解,并且最初也易于构建。此外,对于各种图形化开发工具,入门也很容易。
email
的值。 email
地址。 email
地址有效:
email
地址无效:
join.jsp
。 email
域。
|
|
在清单 1 中,不是 Java 代码中有大量的 HTML,而是在 HTML 文件中有大量的 Java 代码。从这个观点来看,除了允许网页设计人员编写 Java 代码之外,我实际上没做什么。但是,我们并不是一无所有;在 JSP 1.1 中,我们获得一种称为“标记”的新特性。
JSP 标记只是将代码从 JSP 文件中抽取出来的一种方式。有人将 JSP 标记看作是 JSP 文件的宏,其中用于这个标记的代码包含在 servlet 中。(宏的观点在很大程度上是正确的。)出于同样的原因,我不希望在 Java 代码中看到 HTML 标记,我也不希望在 JSP 文件中看到 Java 代码。JSP 技术的整个出发点就是允许网页设计人员创建 servlet,而不必纠缠于 Java 代码。标记允许 Java 程序员将 Java 代码伪装成 HTML 来扩展 JSP 文件。图 3 显示了从 JSP 网页中抽取代码并将它们放入 JSP 标记中的一般概念。
清单 2 是用来说明 Struts 标记的功能的一个例子。在清单 2 中,正常的 HTML <form>
标记被用 Struts <form:form>
标记替换。清单 3 显示了浏览器接收到的结果 HTML。浏览器获得 HTML <form>
标记,但带有附加代码,如 JavaScript。附加的 JavaScript 激活 email
地址域。服务器端的 <form:form>
标记代码创建适当的 HTML,并使网页设计人员不再接触 JavaScript。
<form:form action="join.do" focus="email" > <form:text property="email" size="30" maxlength="30"/> <form:submit property="submit" value="Submit"/> </form:form> |
<form name="joinForm" method="POST" action="join.do;jsessionid=ndj71hjo01"> <input type="text" name="email" maxlength="30" size="30" value=""> <input type="submit" name="submit" value="Submit"> </form> <script language="JavaScript"> <!-- document.joinForm.email.focus() // --> </script> |
include
的 JSP 机制将 HTML 和 JavaScript 添加到网页中。但是,开发人员常常会创建巨大的 JavaScript 库文件,这些库文件被包含在 JSP 文件中。结果返回给客户机的 HTML 网页要比必需的 HMTL 网页大得多。 include
的正确用法是仅将它用于生成诸如页眉和页脚这类内容的 HTML 代码段。
|
|
JSP 标记只解决了部分问题。我们还得处理验证、流程控制和更新应用程序的状态等问题。这正是 MVC 发挥作用的地方。MVC 通过将问题分为三个类别来帮助解决单一模块方法所遇到的某些问题:
|
|
Web 向软件开发人员提出了一些特有的挑战,最明显的就是客户机和服务器的无状态连接。这种无状态行为使得模型很难将更改通知视图。在 Web 上,为了发现对应用程序状态的修改,浏览器必须重新查询服务器。
另一个重大变化是实现视图所用的技术与实现模型或控制器的技术不同。当然,我们可以使用 Java(或者 PERL、C/C++ 或别的语言)代码生成 HTML。这种方法有几个缺点:
对于 Web,需要修改标准的 MVC 形式。图 4 显示了 MVC 的 Web 改写版,通常也称为 MVC Model 2 或 MVC 2。
|
|
Struts 是一组相互协作的类、servlet 和 JSP 标记,它们组成一个可重用的 MVC 2 设计。这个定义表示 Struts 是一个框架,而不是一个库,但 Struts 也包含了丰富的标记库和独立于该框架工作的实用程序类。图 5 显示了 Struts 的一个概览。
struts-config.xml
文件配置控制器。 Action
类完成的。
|
|
图 6 显示的是 org.apache.struts.action
包的一个最简 UML 图。图 6 显示了 ActionServlet
(Controller)、 ActionForm
(Form State) 和 Action
(Model Wrapper) 之间的最简关系。
图 6. Command (ActionServlet) 与 Model (Action & ActionForm) 之间的关系的 UML 图
您还记得函数映射的日子吗?在那时,您会将某些输入事件映射到一个函数指针上。如果您对此比较熟悉,您会将配置信息放入一个文件,并在运行时加载这个文件。函数指针数组曾经是用 C 语言进行结构化编程的很好方法。
现在好多了,我们有了 Java 技术、XML、J2EE,等等。Struts 的控制器是将事件(事件通常是 HTTP post)映射到类的一个 servlet。正如您所料 -- 控制器使用配置文件以使您不必对这些值进行硬编码。时代变了,但方法依旧。
ActionServlet
是该 MVC 实现的 Command 部分,它是这一框架的核心。 ActionServlet
(Command) 创建并使用 Action
、 ActionForm
和 ActionForward
。如前所述, struts-config.xml
文件配置该 Command。在创建 Web 项目时,您将扩展 Action
和 ActionForm
来解决特定的问题。文件 struts-config.xml
指示 ActionServlet
如何使用这些扩展的类。这种方法有几个优点:
可以通过扩展 ActionServlet
来添加 Command 功能。
ActionForm
维护 Web 应用程序的会话状态。 ActionForm
是一个抽象类,必须为每个输入表单模型创建该类的子类。当我说 输入表单模型 时,是指 ActionForm
表示的是由 HTML 表单设置或更新的一般意义上的数据。例如,您可能有一个由 HTML 表单设置的 UserActionForm
。Struts 框架将执行以下操作:
UserActionForm
是否存在;如果不存在,它将创建该类的一个实例。 UserActionForm
的状态。没有太多讨厌的 request.getParameter()
调用。例如,Struts 框架将从请求流中提取 fname
,并调用 UserActionForm.setFname()
。 UserActionForm
传递给业务包装 UserAction
之前将更新它的状态。 Action
类之前,Struts 还会对 UserActionForm
调用 validation()
方法进行表单状态验证。 注: 这并不总是明智之举。别的网页或业务可能使用 UserActionForm
,在这些地方,验证可能有所不同。在 UserAction
类中进行状态验证可能更好。 UserActionForm
。 注:
struts-config.xml
文件控制 HTML 表单请求与 ActionForm
之间的映射关系。 UserActionForm
。 UserActionForm
可跨多页进行映射,以执行诸如向导之类的操作。 Action
类是业务逻辑的一个包装。 Action
类的用途是将 HttpServletRequest
转换为业务逻辑。要使用 Action
,请创建它的子类并覆盖 process()
方法。
ActionServlet
(Command) 使用 perform()
方法将参数化的类传递给 ActionForm
。仍然没有太多讨厌的 request.getParameter()
调用。当事件进展到这一步时,输入表单数据(或 HTML 表单数据)已被从请求流中提取出来并转移到 ActionForm
类中。
注:扩展 Action
类时请注意简洁。 Action
类应该控制应用程序的流程,而不应该控制应用程序的逻辑。通过将业务逻辑放在单独的包或 EJB 中,我们就可以提供更大的灵活性和可重用性。
考虑 Action
类的另一种方式是 Adapter 设计模式。 Action
的用途是“将类的接口转换为客户机所需的另一个接口。Adapter 使类能够协同工作,如果没有 Adapter,则这些类会因为不兼容的接口而无法协同工作。”(摘自 Gof 所著的 Design Patterns - Elements of Reusable OO Software )。本例中的客户机是 ActionServlet
,它对我们的具体业务类接口一无所知。因此,Struts 提供了它能够理解的一个业务接口,即 Action
。通过扩展 Action
,我们使得我们的业务接口与 Struts 业务接口保持兼容。(一个有趣的发现是, Action
是类而不是接口)。 Action
开始为一个接口,后来却变成了一个类。真是金无足赤。)
UML 图(图 6)还包括 ActionError
和 ActionErrors
。 ActionError
封装了单个错误消息。 ActionErrors
是 ActionError
类的容器,View 可以使用标记访问这些类。 ActionError
是 Struts 保持错误列表的方式。
图 7. Command (ActionServlet) 与 Model (Action) 之间的关系的 UML 图
输入事件通常是在 HTTP 请求表单中发生的,servlet 容器将 HTTP 请求转换为 HttpServletRequest
。控制器查看输入事件并将请求分派给某个 Action
类。 struts-config.xml
确定 Controller 调用哪个 Action
类。 struts-config.xml
配置信息被转换为一组 ActionMapping
,而后者又被放入 ActionMappings
容器中。(您可能尚未注意到这一点,以 s 结尾的类就是容器)
ActionMapping
包含有关特定事件如何映射到特定 Action
的信息。 ActionServlet
(Command) 通过 perform()
方法将 ActionMapping
传递给 Action
类。这样就使 Action
可访问用于控制流程的信息。
ActionMappings
是 ActionMapping
对象的一个集合。
|
|
下面我们看一下 Struts 是如何解决困扰 join.jsp
的这些问题的。改写后的方案由两个项目组成。第一个项目包含应用程序的逻辑部分,这个应用程序是独立于 Web 应用程序的。这个独立层可能是用 EJB 技术实现的公共服务层。为了便于说明,我使用 Ant 构建进程创建了一个称为 business
的包。有几个原因促使我们使用独立的业务层:
我用 Ant 构建项目,并用 JUnit 运行单元测试。business.zip 包含构建业务项目所需的一切,当然 Ant 和 JUnit 除外。这个包脚本将构建类,运行单元测试,创建 Java 文档和 jar 文件,最后将所有这些内容压缩到一个 zip 文件中发送给客户。只要对 build.xml
作一些修改,您就可以将它部署到其他平台上。 Business.jar
位于 Web 的下载部分,因此,您并非必须下载并构建这个业务包。
第二个项目是用 Struts 开发的一个 Web 应用程序。您将需要一个符合 JSP 1.1 和 Servlet 2.2 规范的容器。最快的入门方法是下载并安装 Tomcat 3.2(请参阅 参考资源 )。直到有 Struts 的 1.0 发行版之前,我建议您从 Jakarta 项目获得最新的版本(请参阅 参考资源 )。这对我来说是个大问题,我不能确保我的 Web 项目样例能与您下载的 Struts 一起工作。Struts 仍在不断变化,所以我不得不经常更新我的项目。在本项目中,我使用的是 jakarta-struts-20010105.zip。图 8 显示了此 Web 项目的结构。如果您已安装了 Ant,则运行这个版本将创建一个称为 joinStruts.war
的 war 文件,您随时可以部署这个文件。
清单 4 显示了转换后的 JSP 文件,称为 joinMVC.jsp
。这个文件从最初的 50 行变为 19 行,并且现在不含任何 Java 代码。从网页设计人员的角度来看,这是个巨大的改进。
清单 4. joinMVC.jsp -- 再访简单的 JSP
<%@ page language="java" %> <%@ taglib uri="/WEB-INF/struts.tld" prefix="struts" %> <%@ taglib uri="/WEB-INF/struts-form.tld" prefix="form" %> <html> <head> <title><struts:message key="join.title"/></title> </head> <body bgcolor="white"> <form:errors/> <h3>Enter your email to join the group</h3> <form:form action="join.do" focus="email" > <form:text property="email" size="30" maxlength="30"/> <form:submit property="submit" value="Submit"/> </form:form> </body> </html> |
下面是使用 Struts 标记库之后所发生变化的列表:
<%@ taglib uri="/WEB-INF/struts.tld" prefix="struts" %> |
<%@page import?
已被替换为用于 Struts 标记库的 <%@ taglib uri?
。 <struts:message key="join.title"/> |
join.title
的文本。在本例中,ApplicationResources 属性文件包含这个名值对。这使字符串更易于查看和国际化。 <form:errors/> |
ActionServlet
或 ActionForm
构建要显示的错误消息。这些错误消息也可以包含在属性文件中。ApplicationResources 也提供了一种格式化错误的方法,即设置 error.header
和 error.footer
。 <form:form action="join.do" focus="email" > |
<form>
标记和属性替代了 HTML <form>
标记和属性。 <form action="join.jsp" name="join">
已更改为 <form:form action="join.do" focus="email" >
。 <input>
标记已替换为 <form:text/>
。 <submit>
标记已替换为 <form:submit/>
。 JoinForm
扩展了 ActionForm
并包含表单数据。本例中的表单数据只有电子邮件地址。我已为电子邮件地址添加了一个写方法和读方法,以供框架访问。为了便于说明,我重写了 validate()
方法,并使用了 Struts 的跟踪功能。Struts 将创建 JoinForm
并设置状态信息。
如前所述, Action
是控制器和实际业务对象之间的接口。 JoinAction
包装了对 business.jar
的调用,这些调用最初在 join.jsp
文件中。 JoinAction
的 perform()
方法在清单 5 中列表。
public ActionForward perform(ActionMapping mapping, ActionForm form, HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException { // 抽取我们将会用到的属性和参数 JoinForm joinForm = (JoinForm) form; String email = joinForm.getEmail(); ActionErrors errors = new ActionErrors(); // 存储输入.... try { business.db.MailingList.AddEmail(email); } catch (Exception e) { // 记录日志,打印栈 // 将错误回显给用户 errors.add("email",new ActionError("error.mailing.db.add")); } // 如需任何消息,请将指定的错误消息键保存到 // HTTP 请求中,以供 <struts:errors> 标记使用。 if (!errors.empty()) { saveErrors(request, errors); // 返回到初始表单 return (new ActionForward(mapping.getInput())); } // 将控制权转交给 Action.xml 中指定的 'success' URI return (mapping.findForward("success")); } |
注: perform()
返回一个称为 ActionForward
的类,该类通知控制器下一步该执行什么操作。在本例中,我使用从控制器传入的映射来决定下一步的操作。
我已修改了 JSP 文件,并创建了两个新类:一个类用来包含表单数据,一个类用来调用业务包。最后,我通过修改配置文件 struts-config.xml
将它们整合起来。清单 6 显示了我添加的 action 元素,这个元素用来控制 joinMVC.jsp
的流程。
<action path="/join" name="joinForm" type="web.mailinglist.JoinAction" scope="request" input="/joinMVC.jsp" validate="true"> <forward name="success" path="/welcome.html"/> </action> |
action
元素描述了从请求路径到相应的 Action 类的映射,应该用这些类来处理来自这个路径的请求。每个请求类型都应该有相应的 action
元素,用来描述如何处理该请求。对于 join 请求:
joinForm
用来容纳表单数据。 joinForm
将试图进行自我验证。 web.mailinglist.JoinAction
是用来处理对这个映射的请求的 action 类。 welcome.jsp
。 joinMVC.jsp
,这是最初发出请求的网页。为什么会这样呢?在清单 6 的 action 元素中,有一个称为 input
的属性,其值为 "/joinMVC.jsp"
。在我的 JoinAction.perform()
(如清单 5 所示)中,如果业务逻辑失败, perform()
就返回一个 ActionForward
,并以 mapping.getInput()
作为参数。本例中的 getInput()
是 "/joinMVC.jsp"
。如果业务逻辑失败,它将返回到 joinMVC.jsp
,这是最初发出请求的网页。
|
|
正如我们在图 9 中所看到的那样,复杂性和层都有显著增加。不再存在从 JSP 文件到 Service 层的直接调用。