Struts,MVC 的一种开放源码实现用这种servlet和JSP框架管理复杂的大型网站

news/2024/5/21 1:02:33/文章来源:https://blog.csdn.net/pharaohsprince/article/details/479515
(转载自IBM developerWorks中国网站 java 专区,http://www-900.ibm.com/developerWorks/java/j-struts/index.shtml)

Malcolm Davis
顾问
2001 年 2 月

内容:
 简介
 JSP 就是 servlet
 别在我的 HTML 中使用太多的 Java
 MVC
 MVC Model 2
 Struts
 详细分析 Struts
 邮件列表样例
 前后比较
 Struts 的前景
 最后的注释
 参考资源
 作者简介

本文介绍 Struts,它是使用 servlet 和 JavaServer Pages 技术的一种 Model-View-Controller 实现。Struts 可帮助您控制 Web 项目中的变化并提高专业化水平。尽管您可能永远不会用 Struts 实现一个系统,但您可以将其中的一些思想用于您以后的 servlet 和 JSP 网页的实现中。

简介
小学生也可以在因特网上发布 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 文件,在开发过程中我已顾及到网页设计人员和变化。

一个 JSP 文件就是一个 Java servlet
JavaServer Page (JSP) 文件只是审视 servlet 的另一种方式。JSP 文件的概念使我们能够将 Java servlet 看作一个 HTML 网页。JSP 消除了 Java 代码中经常出现的讨厌的 print() 语句。JSP 文件首先被预处理为 .java 文件,然后再编译为 .class 文件。如果您使用的是 Tomcat,则可以在 work 目录下查看预处理后的 .java 文件。别的容器可能将 .java.class 文件存储在其他位置;这个位置与容器有关。图 1 说明了从 JSP 文件到 servlet 的流程。

图 1. 从 JSP 文件到 servlet 的流程
从 JSP 到 servlet 的流程

(这与 Microsoft 的 Active Server Page (ASP) 明显不同。ASP 被编译到内存中,而不是编译到一个单独的文件中。)

简单的独立 JSP 文件
在小型 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 文件是一个独立的、自主完成所有任务的模块。未包含在这个 JSP 文件中的仅有代码是包含在 isValidEmail() 中的实际验证代码和将电子邮件地址存入数据库的代码。(将 isValidEmail() 方法分离到可重用的代码中似乎是当然的选择,但我曾见过直接嵌入网页中的 isValidEmail() 代码。单页方法的优点是易于理解,并且最初也易于构建。此外,对于各种图形化开发工具,入门也很容易。

join.jsp 的活动

  1. 显示打开的输入网页。
  2. 从表单参数中读取 email 的值。
  3. 验证 email 地址。
  4. 如果 email 地址有效:
    • 将该地址添加到数据库中。
    • 重定向到下一个网页。
  5. 如果 email 地址无效:
    • 设置错误消息。
    • 重新显示含有错误消息的 join.jsp

单页方法的后果

  • HTML 和 Java 强耦合在一起
    JSP 文件的编写者必须既是网页设计者,又是 Java 开发者。其结果通常要么是很糟的 Java 代码,要么是难看的网页,有时甚至 Java 代码和网页都很糟。

  • Java 和 JavaScript 的不足
    随着网页逐渐变大,很容易想到实现一些 JavaScript。当网页中出现 JavaScript 时,这种脚本就可能与 Java 代码产生混淆。可能产生混淆的一个例子是使用客户端的 JavaScript 来验证 email 域。

  • 内嵌的流程逻辑
    要理解应用程序的整个流程,您必须浏览所有网页。试想一下拥有 100 个网页的网站的错综复杂的逻辑。

  • 调试困难
    除了很糟的外观之外,HTML 标记、Java 代码和 JavaScript 代码都集中在一个网页中还使调试变得相当困难。

  • 强耦合
    更改业务逻辑或数据可能牵涉相关的每个网页。

  • 美学
    在很大的网页中,这编码样式看起来杂乱无章。我过去进行 Microsoft ASP 开发时,我经常看到有 1000 行的网页。即使有彩色语法显示,阅读和理解这些代码仍然比较困难。

图 1. 从 JSP 文件到 servlet 的流程
从 JSP 到 servlet 的流程

(这与 Microsoft 的 Active Server Page (ASP) 明显不同。ASP 被编译到内存中,而不是编译到一个单独的文件中。)

简单的独立 JSP 文件
在小型 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 文件是一个独立的、自主完成所有任务的模块。未包含在这个 JSP 文件中的仅有代码是包含在 isValidEmail() 中的实际验证代码和将电子邮件地址存入数据库的代码。(将 isValidEmail() 方法分离到可重用的代码中似乎是当然的选择,但我曾见过直接嵌入网页中的 isValidEmail() 代码。单页方法的优点是易于理解,并且最初也易于构建。此外,对于各种图形化开发工具,入门也很容易。

join.jsp 的活动

  1. 显示打开的输入网页。
  2. 从表单参数中读取 email 的值。
  3. 验证 email 地址。
  4. 如果 email 地址有效:
    • 将该地址添加到数据库中。
    • 重定向到下一个网页。
  5. 如果 email 地址无效:
    • 设置错误消息。
    • 重新显示含有错误消息的 join.jsp

单页方法的后果

  • HTML 和 Java 强耦合在一起
    JSP 文件的编写者必须既是网页设计者,又是 Java 开发者。其结果通常要么是很糟的 Java 代码,要么是难看的网页,有时甚至 Java 代码和网页都很糟。

  • Java 和 JavaScript 的不足
    随着网页逐渐变大,很容易想到实现一些 JavaScript。当网页中出现 JavaScript 时,这种脚本就可能与 Java 代码产生混淆。可能产生混淆的一个例子是使用客户端的 JavaScript 来验证 email 域。

  • 内嵌的流程逻辑
    要理解应用程序的整个流程,您必须浏览所有网页。试想一下拥有 100 个网页的网站的错综复杂的逻辑。

  • 调试困难
    除了很糟的外观之外,HTML 标记、Java 代码和 JavaScript 代码都集中在一个网页中还使调试变得相当困难。

  • 强耦合
    更改业务逻辑或数据可能牵涉相关的每个网页。

  • 美学
    在很大的网页中,这编码样式看起来杂乱无章。我过去进行 Microsoft ASP 开发时,我经常看到有 1000 行的网页。即使有彩色语法显示,阅读和理解这些代码仍然比较困难。

图 2. 在简单的请求和响应中,JSP 文件设置数据、控制到下一个网页的流程并创建 HTML
简单的请求和响应 JSP

这个邮件列表 JSP 文件是一个独立的、自主完成所有任务的模块。未包含在这个 JSP 文件中的仅有代码是包含在 isValidEmail() 中的实际验证代码和将电子邮件地址存入数据库的代码。(将 isValidEmail() 方法分离到可重用的代码中似乎是当然的选择,但我曾见过直接嵌入网页中的 isValidEmail() 代码。单页方法的优点是易于理解,并且最初也易于构建。此外,对于各种图形化开发工具,入门也很容易。

join.jsp 的活动

  1. 显示打开的输入网页。
  2. 从表单参数中读取 email 的值。
  3. 验证 email 地址。
  4. 如果 email 地址有效:
    • 将该地址添加到数据库中。
    • 重定向到下一个网页。
  5. 如果 email 地址无效:
    • 设置错误消息。
    • 重新显示含有错误消息的 join.jsp

单页方法的后果

  • HTML 和 Java 强耦合在一起
    JSP 文件的编写者必须既是网页设计者,又是 Java 开发者。其结果通常要么是很糟的 Java 代码,要么是难看的网页,有时甚至 Java 代码和网页都很糟。

  • Java 和 JavaScript 的不足
    随着网页逐渐变大,很容易想到实现一些 JavaScript。当网页中出现 JavaScript 时,这种脚本就可能与 Java 代码产生混淆。可能产生混淆的一个例子是使用客户端的 JavaScript 来验证 email 域。

  • 内嵌的流程逻辑
    要理解应用程序的整个流程,您必须浏览所有网页。试想一下拥有 100 个网页的网站的错综复杂的逻辑。

  • 调试困难
    除了很糟的外观之外,HTML 标记、Java 代码和 JavaScript 代码都集中在一个网页中还使调试变得相当困难。

  • 强耦合
    更改业务逻辑或数据可能牵涉相关的每个网页。

  • 美学
    在很大的网页中,这编码样式看起来杂乱无章。我过去进行 Microsoft ASP 开发时,我经常看到有 1000 行的网页。即使有彩色语法显示,阅读和理解这些代码仍然比较困难。

请别在我的 HTML 中加入太多的 Java 代码
在清单 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 标记中的一般概念。

图 3. JSP 标记
JSP 标记分解

清单 2 是用来说明 Struts 标记的功能的一个例子。在清单 2 中,正常的 HTML <form> 标记被用 Struts <form:form> 标记替换。清单 3 显示了浏览器接收到的结果 HTML。浏览器获得 HTML <form> 标记,但带有附加代码,如 JavaScript。附加的 JavaScript 激活 email 地址域。服务器端的 <form:form> 标记代码创建适当的 HTML,并使网页设计人员不再接触 JavaScript。

清单 2. Struts 的 form 标记

<form:form action="join.do" focus="email" >
<form:text   property="email" size="30" maxlength="30"/>
<form:submit property="submit" value="Submit"/>
</form:form>

清单 3. 发送给浏览器的结果 HTML

<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>

有关 JSP 标记的注意事项:

  • JSP 标记需要一个运行 JSP 1.1 或更高版本的容器。

  • JSP 标记在服务器上运行,而不像 HTML 标记那样由客户机解释。

  • JSP 标记提供了适当的代码重用机制。

  • 可以使用一种称为 include 的 JSP 机制将 HTML 和 JavaScript 添加到网页中。但是,开发人员常常会创建巨大的 JavaScript 库文件,这些库文件被包含在 JSP 文件中。结果返回给客户机的 HTML 网页要比必需的 HMTL 网页大得多。include 的正确用法是仅将它用于生成诸如页眉和页脚这类内容的 HTML 代码段。

  • 通过抽取出 Java 代码,JSP 标记使开发角色更加专业化。

清单 2 是用来说明 Struts 标记的功能的一个例子。在清单 2 中,正常的 HTML <form> 标记被用 Struts <form:form> 标记替换。清单 3 显示了浏览器接收到的结果 HTML。浏览器获得 HTML <form> 标记,但带有附加代码,如 JavaScript。附加的 JavaScript 激活 email 地址域。服务器端的 <form:form> 标记代码创建适当的 HTML,并使网页设计人员不再接触 JavaScript。

清单 2. Struts 的 form 标记

<form:form action="join.do" focus="email" >
<form:text   property="email" size="30" maxlength="30"/>
<form:submit property="submit" value="Submit"/>
</form:form>

清单 3. 发送给浏览器的结果 HTML

<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>

有关 JSP 标记的注意事项:

  • JSP 标记需要一个运行 JSP 1.1 或更高版本的容器。

  • JSP 标记在服务器上运行,而不像 HTML 标记那样由客户机解释。

  • JSP 标记提供了适当的代码重用机制。

  • 可以使用一种称为 include 的 JSP 机制将 HTML 和 JavaScript 添加到网页中。但是,开发人员常常会创建巨大的 JavaScript 库文件,这些库文件被包含在 JSP 文件中。结果返回给客户机的 HTML 网页要比必需的 HMTL 网页大得多。include 的正确用法是仅将它用于生成诸如页眉和页脚这类内容的 HTML 代码段。

  • 通过抽取出 Java 代码,JSP 标记使开发角色更加专业化。

模型-视图-控制器 (MVC)
JSP 标记只解决了部分问题。我们还得处理验证、流程控制和更新应用程序的状态等问题。这正是 MVC 发挥作用的地方。MVC 通过将问题分为三个类别来帮助解决单一模块方法所遇到的某些问题:

  • Model(模型)
    模型包含应用程序的核心功能。模型封装了应用程序的状态。有时它包含的唯一功能就是状态。它对视图或控制器一无所知。

  • View(视图)
    视图提供模型的表示。它是应用程序的外观。视图可以访问模型的读方法,但不能访问写方法。此外,它对控制器一无所知。当更改模型时,视图应得到通知。

  • Controller(控制器)
    控制器对用户的输入作出反应。它创建并设置模型。

MVC Model 2
Web 向软件开发人员提出了一些特有的挑战,最明显的就是客户机和服务器的无状态连接。这种无状态行为使得模型很难将更改通知视图。在 Web 上,为了发现对应用程序状态的修改,浏览器必须重新查询服务器。

另一个重大变化是实现视图所用的技术与实现模型或控制器的技术不同。当然,我们可以使用 Java(或者 PERL、C/C++ 或别的语言)代码生成 HTML。这种方法有几个缺点:

  • Java 程序员应该开发服务,而不是 HTML。
  • 更改布局时需要更改代码。
  • 服务的用户应该能够创建网页来满足它们的特定需要。
  • 网页设计人员不能直接参与网页开发。
  • 嵌在代码中的 HTML 很难看。

对于 Web,需要修改标准的 MVC 形式。图 4 显示了 MVC 的 Web 改写版,通常也称为 MVC Model 2 或 MVC 2。

图 4. MVC Model 2
MVC Model 2

Struts,MVC 2 的一种实现
Struts 是一组相互协作的类、servlet 和 JSP 标记,它们组成一个可重用的 MVC 2 设计。这个定义表示 Struts 是一个框架,而不是一个库,但 Struts 也包含了丰富的标记库和独立于该框架工作的实用程序类。图 5 显示了 Struts 的一个概览。

图 5. Struts 概览
Struts 概览

Struts 概览

  • Client browser(客户浏览器)
    来自客户浏览器的每个 HTTP 请求创建一个事件。Web 容器将用一个 HTTP 响应作出响应。

  • Controller(控制器)
    控制器接收来自浏览器的请求,并决定将这个请求发往何处。就 Struts 而言,控制器是以 servlet 实现的一个命令设计模式。struts-config.xml 文件配置控制器。

  • 业务逻辑
    业务逻辑更新模型的状态,并帮助控制应用程序的流程。就 Struts 而言,这是通过作为实际业务逻辑“瘦”包装的 Action 类完成的。

  • Model(模型)的状态
    模型表示应用程序的状态。业务对象更新应用程序的状态。ActionForm bean 在会话级或请求级表示模型的状态,而不是在持久级。JSP 文件使用 JSP 标记读取来自 ActionForm bean 的信息。

  • View(视图)
    视图就是一个 JSP 文件。其中没有流程逻辑,没有业务逻辑,也没有模型信息 -- 只有标记。标记是使 Struts 有别于其他框架(如 Velocity)的因素之一。

Struts 概览

  • Client browser(客户浏览器)
    来自客户浏览器的每个 HTTP 请求创建一个事件。Web 容器将用一个 HTTP 响应作出响应。

  • Controller(控制器)
    控制器接收来自浏览器的请求,并决定将这个请求发往何处。就 Struts 而言,控制器是以 servlet 实现的一个命令设计模式。struts-config.xml 文件配置控制器。

  • 业务逻辑
    业务逻辑更新模型的状态,并帮助控制应用程序的流程。就 Struts 而言,这是通过作为实际业务逻辑“瘦”包装的 Action 类完成的。

  • Model(模型)的状态
    模型表示应用程序的状态。业务对象更新应用程序的状态。ActionForm bean 在会话级或请求级表示模型的状态,而不是在持久级。JSP 文件使用 JSP 标记读取来自 ActionForm bean 的信息。

  • View(视图)
    视图就是一个 JSP 文件。其中没有流程逻辑,没有业务逻辑,也没有模型信息 -- 只有标记。标记是使 Struts 有别于其他框架(如 Velocity)的因素之一。

详细分析 Struts
图 6 显示的是 org.apache.struts.action 包的一个最简 UML 图。图 6 显示了 ActionServlet (Controller)、ActionForm (Form State) 和 Action (Model Wrapper) 之间的最简关系。

图 6. Command (ActionServlet) 与 Model (Action & ActionForm) 之间的关系的 UML 图
ActionServlet 与 Action 和 ActionForm 的关系

ActionServlet
您还记得函数映射的日子吗?在那时,您会将某些输入事件映射到一个函数指针上。如果您对此比较熟悉,您会将配置信息放入一个文件,并在运行时加载这个文件。函数指针数组曾经是用 C 语言进行结构化编程的很好方法。

现在好多了,我们有了 Java 技术、XML、J2EE,等等。Struts 的控制器是将事件(事件通常是 HTTP post)映射到类的一个 servlet。正如您所料 -- 控制器使用配置文件以使您不必对这些值进行硬编码。时代变了,但方法依旧。

ActionServlet 是该 MVC 实现的 Command 部分,它是这一框架的核心。ActionServlet (Command) 创建并使用 ActionActionFormActionForward。如前所述,struts-config.xml 文件配置该 Command。在创建 Web 项目时,您将扩展 ActionActionForm 来解决特定的问题。文件 struts-config.xml 指示 ActionServlet 如何使用这些扩展的类。这种方法有几个优点:

  • 应用程序的整个逻辑流程都存储在一个分层的文本文件中。这使得人们更容易查看和理解它,尤其是对于大型应用程序而言。

  • 网页设计人员不必费力地阅读 Java 代码来理解应用程序的流程。

  • Java 开发人员也不必在更改流程以后重新编译代码。

可以通过扩展 ActionServlet 来添加 Command 功能。

ActionForm

ActionForm 维护 Web 应用程序的会话状态。ActionForm 是一个抽象类,必须为每个输入表单模型创建该类的子类。当我说输入表单模型时,是指 ActionForm 表示的是由 HTML 表单设置或更新的一般意义上的数据。例如,您可能有一个由 HTML 表单设置的 UserActionForm。Struts 框架将执行以下操作:

  • 检查 UserActionForm 是否存在;如果不存在,它将创建该类的一个实例。

  • Struts 将使用 HttpServletRequest 中相应的域设置 UserActionForm 的状态。没有太多讨厌的 request.getParameter() 调用。例如,Struts 框架将从请求流中提取 fname,并调用 UserActionForm.setFname()

  • Struts 框架在将 UserActionForm 传递给业务包装 UserAction 之前将更新它的状态。

  • 在将它传递给 Action 类之前,Struts 还会对 UserActionForm 调用 validation() 方法进行表单状态验证。注:这并不总是明智之举。别的网页或业务可能使用 UserActionForm,在这些地方,验证可能有所不同。在 UserAction 类中进行状态验证可能更好。

  • 可在会话级维护 UserActionForm

注:

  • struts-config.xml 文件控制 HTML 表单请求与 ActionForm 之间的映射关系。
  • 可将多个请求映射到 UserActionForm
  • UserActionForm 可跨多页进行映射,以执行诸如向导之类的操作。

Action
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 开始为一个接口,后来却变成了一个类。真是金无足赤。)

Error
UML 图(图 6)还包括 ActionErrorActionErrorsActionError 封装了单个错误消息。ActionErrorsActionError 类的容器,View 可以使用标记访问这些类。ActionError 是 Struts 保持错误列表的方式。

图 7. Command (ActionServlet) 与 Model (Action) 之间的关系的 UML 图
ActionServlet 与 Action 的关系

ActionMapping
输入事件通常是在 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
ActionMappingsActionMapping 对象的一个集合。

图 6. Command (ActionServlet) 与 Model (Action & ActionForm) 之间的关系的 UML 图
ActionServlet 与 Action 和 ActionForm 的关系

ActionServlet
您还记得函数映射的日子吗?在那时,您会将某些输入事件映射到一个函数指针上。如果您对此比较熟悉,您会将配置信息放入一个文件,并在运行时加载这个文件。函数指针数组曾经是用 C 语言进行结构化编程的很好方法。

现在好多了,我们有了 Java 技术、XML、J2EE,等等。Struts 的控制器是将事件(事件通常是 HTTP post)映射到类的一个 servlet。正如您所料 -- 控制器使用配置文件以使您不必对这些值进行硬编码。时代变了,但方法依旧。

ActionServlet 是该 MVC 实现的 Command 部分,它是这一框架的核心。ActionServlet (Command) 创建并使用 ActionActionFormActionForward。如前所述,struts-config.xml 文件配置该 Command。在创建 Web 项目时,您将扩展 ActionActionForm 来解决特定的问题。文件 struts-config.xml 指示 ActionServlet 如何使用这些扩展的类。这种方法有几个优点:

  • 应用程序的整个逻辑流程都存储在一个分层的文本文件中。这使得人们更容易查看和理解它,尤其是对于大型应用程序而言。

  • 网页设计人员不必费力地阅读 Java 代码来理解应用程序的流程。

  • Java 开发人员也不必在更改流程以后重新编译代码。

可以通过扩展 ActionServlet 来添加 Command 功能。

ActionForm

ActionForm 维护 Web 应用程序的会话状态。ActionForm 是一个抽象类,必须为每个输入表单模型创建该类的子类。当我说输入表单模型时,是指 ActionForm 表示的是由 HTML 表单设置或更新的一般意义上的数据。例如,您可能有一个由 HTML 表单设置的 UserActionForm。Struts 框架将执行以下操作:

  • 检查 UserActionForm 是否存在;如果不存在,它将创建该类的一个实例。

  • Struts 将使用 HttpServletRequest 中相应的域设置 UserActionForm 的状态。没有太多讨厌的 request.getParameter() 调用。例如,Struts 框架将从请求流中提取 fname,并调用 UserActionForm.setFname()

  • Struts 框架在将 UserActionForm 传递给业务包装 UserAction 之前将更新它的状态。

  • 在将它传递给 Action 类之前,Struts 还会对 UserActionForm 调用 validation() 方法进行表单状态验证。注:这并不总是明智之举。别的网页或业务可能使用 UserActionForm,在这些地方,验证可能有所不同。在 UserAction 类中进行状态验证可能更好。

  • 可在会话级维护 UserActionForm

注:

  • struts-config.xml 文件控制 HTML 表单请求与 ActionForm 之间的映射关系。
  • 可将多个请求映射到 UserActionForm
  • UserActionForm 可跨多页进行映射,以执行诸如向导之类的操作。

Action
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 开始为一个接口,后来却变成了一个类。真是金无足赤。)

Error
UML 图(图 6)还包括 ActionErrorActionErrorsActionError 封装了单个错误消息。ActionErrorsActionError 类的容器,View 可以使用标记访问这些类。ActionError 是 Struts 保持错误列表的方式。

图 7. Command (ActionServlet) 与 Model (Action) 之间的关系的 UML 图
ActionServlet 与 Action 的关系

ActionMapping
输入事件通常是在 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
ActionMappingsActionMapping 对象的一个集合。

图 7. Command (ActionServlet) 与 Model (Action) 之间的关系的 UML 图
ActionServlet 与 Action 的关系

ActionMapping
输入事件通常是在 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
ActionMappingsActionMapping 对象的一个集合。

再访邮件列表样例
下面我们看一下 Struts 是如何解决困扰 join.jsp 的这些问题的。改写后的方案由两个项目组成。第一个项目包含应用程序的逻辑部分,这个应用程序是独立于 Web 应用程序的。这个独立层可能是用 EJB 技术实现的公共服务层。为了便于说明,我使用 Ant 构建进程创建了一个称为 business 的包。有几个原因促使我们使用独立的业务层:

  • 划分责任
    单独的包使管理人员能够在开发小组内委派责任。这也有助于提高开发人员的责任心。

  • 通用件
    我们设想开发人员将这个包看作一个商业软件。将它放在另外的包中使它更像通用件。这个包可能是通用件,也可能是由组织内部的另一个小组开发的。

  • 避免不必要的构建和单元测试。
    分开的构建进程有助于避免不必要的构建和单元测试。

  • 使用接口开发
    在进行开发和避免不必要的耦合时,它有助于从接口的观点来思考问题。这是极重要的一个方面。当开发您自己的业务包时,这些业务类不应该关心到底是 Web 应用程序执行调用,还是独立应用程序执行调用。因此,应该避免在业务逻辑层使用对 servlet API 或 Struts API 调用的任何引用。

  • 稳定性
    并不是每个组织都每天、每周甚至每月进行检修。因此,在进行开发时,稳定的接口点是重要的。不能因为业务包处于变迁阶段就认为 Web 项目也应该处于变迁阶段。

业务构建注释
我用 Ant 构建项目,并用 JUnit 运行单元测试。business.zip 包含构建业务项目所需的一切,当然 Ant 和 JUnit 除外。这个包脚本将构建类,运行单元测试,创建 Java 文档和 jar 文件,最后将所有这些内容压缩到一个 zip 文件中发送给客户。只要对 build.xml 作一些修改,您就可以将它部署到其他平台上。Business.jar 位于 Web 的下载部分,因此,您并非必须下载并构建这个业务包。

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 文件,您随时可以部署这个文件。

图 8. Web 项目的结构
Web 项目的结构

清单 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 标记库之后所发生变化的列表:

模型 -- 会话状态
JoinForm 扩展了 ActionForm 并包含表单数据。本例中的表单数据只有电子邮件地址。我已为电子邮件地址添加了一个写方法和读方法,以供框架访问。为了便于说明,我重写了 validate() 方法,并使用了 Struts 的跟踪功能。Struts 将创建 JoinForm 并设置状态信息。

模型 -- 业务逻辑
如前所述,Action 是控制器和实际业务对象之间的接口。JoinAction 包装了对 business.jar 的调用,这些调用最初在 join.jsp 文件中。JoinActionperform() 方法在清单 5 中列表。

清单 5. - JoinAction.perform()

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 的流程。

清单 6. Action 配置

<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 请求:

  1. joinForm 用来容纳表单数据。
  2. 因为 validate 被标记为 true,所以 joinForm 将试图进行自我验证。
  3. web.mailinglist.JoinAction 是用来处理对这个映射的请求的 action 类。
  4. 如果一切顺利,该请求将转到 welcome.jsp
  5. 如果出现业务逻辑故障,流程将返回到 joinMVC.jsp,这是最初发出请求的网页。为什么会这样呢?在清单 6 的 action 元素中,有一个称为 input 的属性,其值为 "/joinMVC.jsp"。在我的 JoinAction.perform()(如清单 5 所示)中,如果业务逻辑失败,perform() 就返回一个 ActionForward,并以 mapping.getInput() 作为参数。本例中的 getInput()"/joinMVC.jsp"。如果业务逻辑失败,它将返回到 joinMVC.jsp,这是最初发出请求的网页。

清单 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 标记库之后所发生变化的列表:

  • Import
    <%@ taglib uri="/WEB-INF/struts.tld" prefix="struts" %>
    用于 Java 代码的 <%@page import? 已被替换为用于 Struts 标记库的 <%@ taglib uri?

  • 文本
    <struts:message key="join.title"/>
    资源属性文件包含 join.title 的文本。在本例中,ApplicationResources 属性文件包含这个名值对。这使字符串更易于查看和国际化。

  • 错误
    <form:errors/>
    ActionServletActionForm 构建要显示的错误消息。这些错误消息也可以包含在属性文件中。ApplicationResources 也提供了一种格式化错误的方法,即设置 error.headererror.footer

  • HTML 表单
    <form:form action="join.do" focus="email" >
    • JSP <form> 标记和属性替代了 HTML <form> 标记和属性。 <form action="join.jsp" name="join"> 已更改为 <form:form action="join.do" focus="email" >
    • HTML <input> 标记已替换为 <form:text/>
    • HTML <submit> 标记已替换为 <form:submit/>

模型 -- 会话状态
JoinForm 扩展了 ActionForm 并包含表单数据。本例中的表单数据只有电子邮件地址。我已为电子邮件地址添加了一个写方法和读方法,以供框架访问。为了便于说明,我重写了 validate() 方法,并使用了 Struts 的跟踪功能。Struts 将创建 JoinForm 并设置状态信息。

模型 -- 业务逻辑
如前所述,Action 是控制器和实际业务对象之间的接口。JoinAction 包装了对 business.jar 的调用,这些调用最初在 join.jsp 文件中。JoinActionperform() 方法在清单 5 中列表。

清单 5. - JoinAction.perform()

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 的流程。

清单 6. Action 配置

<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 请求:

  1. joinForm 用来容纳表单数据。
  2. 因为 validate 被标记为 true,所以 joinForm 将试图进行自我验证。
  3. web.mailinglist.JoinAction 是用来处理对这个映射的请求的 action 类。
  4. 如果一切顺利,该请求将转到 welcome.jsp
  5. 如果出现业务逻辑故障,流程将返回到 joinMVC.jsp,这是最初发出请求的网页。为什么会这样呢?在清单 6 的 action 元素中,有一个称为 input 的属性,其值为 "/joinMVC.jsp"。在我的 JoinAction.perform()(如清单 5 所示)中,如果业务逻辑失败,perform() 就返回一个 ActionForward,并以 mapping.getInput() 作为参数。本例中的 getInput()"/joinMVC.jsp"。如果业务逻辑失败,它将返回到 joinMVC.jsp,这是最初发出请求的网页。

使用 Struts 前后的比较
正如我们在图 9 中所看到的那样,复杂性和层都有显著增加。不再存在从 JSP 文件到 Service 层的直接调用。

图 9. 使用 Struts 前后的比较
使用 Struts 前后的比较

Struts 的优点

  • JSP 标记机制的使用
    标记特性从 JSP 文件获得可重用代码和抽象 Java 代码。这个特性能很好地集成到基于 JSP 的开发工具中,这些工具允许用标记编写代码。

  • 标记库
    为什么要另发明一种轮子,或标记库呢?如果您在库中找不到您所要的标记,那就自己定义吧。此外,如果您正在学习 JSP 标记技术,则 Struts 为您提供了一个起点。

  • 开放源码
    您可以获得开放源码的全部优点,比如可以查看代码并让使用库的每个人检查代码。许多人都可以进行很好的代码检查。

  • MVC 实现样例
    如果您希望创建您自己的 MVC 实现,则 Struts 可增加您的见识。

  • 管理问题空间
    分治是解决问题并使问题可管理的极好方法。当然,这是一把双刃剑。问题越来越复杂,并且需要越来越多的管理。

Struts 的缺点

  • 仍处于发展初期
    Struts 开发仍处于初级阶段。他们正在向着发行版本 1.0 而努力,但与任何 1.0 版本一样,它不可能尽善尽美。

  • 仍在变化中
    这个框架仍在快速变化。Struts 1.0 与 Struts 0.5 相比变化极大。为了避免使用不赞成使用的方法,您可能隔一天就需要下载最新的 Struts。在过去的 6 个月中,我目睹 Struts 库从 90K 增大到 270K 以上。由于 Struts 中的变化,我不得不数次修改我的示例,但我不保证我的示例能与您下载的 Struts 协同工作。

  • 正确的抽象级别
    Struts 是否提供了正确的抽象级别?对于网页设计人员而言,什么是正确的抽象级别呢?这是一个用 $64K 的文字才能解释清楚的问题。在开发网页的过程中,我们是否应该让网页设计人员访问 Java 代码?某些框架(如 Velocity)说不应该,但它提供了另一种 Web 开发语言让我们学习。在 UI 开发中限制访问 Java 有一定的合理性。最重要的是,如果让网页设计人员使用一点 Java,他将使用大量的 Java。在 Microsoft ASP 的开发中,我总是看到这样的情况。在 ASP 开发中,您应该创建 COM 对象,然后编写少量的 ASP 脚本将这些 COM 对象联系起来。但是,ASP 开发人员会疯狂地使用 ASP 脚本。我会听到这样的话,“既然我可以用 VBScript 直接编写 COM 对象,为什么还要等 COM 开发人员来创建它呢?”通过使用标记库,Struts 有助于限制 JSP 文件中所需的 Java 代码的数量。Logic Tag 就是这样的一种库,它对有条件地生成输出进行管理,但这并不能阻止 UI 开发人员对 Java 代码的狂热。无论您决定使用哪种类型的框架,您都应该了解您要在其中部署和维护该框架的环境。当然,这项任务真是说起来容易做起来难。

  • 有限的适用范围
    Struts 是一种基于 Web 的 MVC 解决方案,所以必须用 HTML、JSP 文件和 servlet 来实现它。

  • J2EE 应用程序支持
    Struts 需要支持 JSP 1.1 和 Servlet 2.2 规范的 servlet 容器。仅凭这一点远不能解决您的全部安装问题,除非使用 Tomcat 3.2。我用 Netscape iPlanet 6.0 安装这个库时遇到一大堆问题,按理说它是第一种符合 J2EE 的应用程序服务器。我建议您在遇到问题时访问 Struts 用户邮件列表的归档资料(请参阅参考资源)。

  • 复杂性
    在将问题分为几个部分的同时也引入了复杂性。毫无疑问,要理解 Struts 必须接受一定的培训。随着变化的不断加入,这有时会令人很沮丧。欢迎访问本网站。

  • 在何处...
    我还能指出其他问题,例如,控制器的客户端验证、可适用工作流程和动态策略模式在什么地方?但是,目前这太容易成为吹毛求疵的问题,有些问题是无关紧要的,或者说应该对 1.0 发行版提这些问题。随着 Struts 小组的不断努力,到您阅读本文时 Struts 说不定已经有了这些功能,或者它很快就会具有这些功能。

图 9. 使用 Struts 前后的比较
使用 Struts 前后的比较

Struts 的优点

  • JSP 标记机制的使用
    标记特性从 JSP 文件获得可重用代码和抽象 Java 代码。这个特性能很好地集成到基于 JSP 的开发工具中,这些工具允许用标记编写代码。

  • 标记库
    为什么要另发明一种轮子,或标记库呢?如果您在库中找不到您所要的标记,那就自己定义吧。此外,如果您正在学习 JSP 标记技术,则 Struts 为您提供了一个起点。

  • 开放源码
    您可以获得开放源码的全部优点,比如可以查看代码并让使用库的每个人检查代码。许多人都可以进行很好的代码检查。

  • MVC 实现样例
    如果您希望创建您自己的 MVC 实现,则 Struts 可增加您的见识。

  • 管理问题空间
    分治是解决问题并使问题可管理的极好方法。当然,这是一把双刃剑。问题越来越复杂,并且需要越来越多的管理。

Struts 的缺点

    • 仍处于发展初期
      Struts 开发仍处于初级阶段。他们正在向着发行版本 1.0 而努力,但与任何 1.0 版本一样,它不可能尽善尽美。

    • 仍在变化中
      这个框架仍在快速变化。Struts 1.0 与 Struts 0.5 相比变化极大。为了避免使用不赞成使用的方法,您可能隔一天就需要下载最新的 Struts。在过去的 6 个月中,我目睹 Struts 库从 90K 增大到 270K 以上。由于 Struts 中的变化,我不得不数次修改我的示例,但我不保证我的示例能与您下载的 Struts 协同工作。

    • 正确的抽象级别
      Struts 是否提供了正确的抽象级别?对于网页设计人员而言,什么是正确的抽象级别呢?这是一个用 $64K 的文字才能解释清楚的问题。在开发网页的过程中,我们是否应该让网页设计人员访问 Java 代码?某些框架(如 Velocity)说不应该,但它提供了另一种 Web 开发语言让我们学习。在 UI 开发中限制访问 Java 有一定的合理性。最重要的是,如果让网页设计人员使用一点 Java,他将使用大量的 Java。在 Microsoft ASP 的开发中,我总是看到这样的情况。在 ASP 开发中,您应该创建 COM 对象,然后编写少量的 ASP 脚本将这些 COM 对象联系起来。但是,ASP 开发人员会疯狂地使用 ASP 脚本。我会听到这样的话,“既然我可以用 VBScript 直接编写 COM 对象,为什么还要等 COM 开发人员来创建它呢?”通过使用标记库,Struts 有助于限制 JSP 文件中所需的 Java 代码的数量。Logic Tag 就是这样的一种库,它对有条件地生成输出进行管理,但这并不能阻止 UI 开发人员对 Java 代码的狂热。无论您决定使用哪种类型的框架,您都应该了解您要在其中部署和维护该框架的环境。当然,这项任务真是说起来容易做起来难。

    • 有限的适用范围
      Struts 是一种基于 Web 的 MVC 解决方案,所以必须用 HTML、JSP 文件和 servlet 来实现它。

  • J2EE 应用程序支持
    Struts 需要支持 JSP 1.1 和 Servlet 2.2 规范的 servlet 容器。仅凭这一点远不能解决您的全部安装问题,除非使用 Tomcat 3.2。我用 Netscape iPlanet 6.0 安装这个库时遇到一大堆问题,按理说它是第一种符合 J2EE 的应用程序服务器。我建议您在遇到问题时访问 Struts 用户邮件列表的归档资料(请参阅参考资源)。

  • 复杂性
    在将问题分为几个部分的同时也引入了复杂性。毫无疑问,要理解 Struts 必须接受一定的培训。随着变化的不断加入,这有时会令人很沮丧。欢迎访问本网站。

  • 在何处...
    我还能指出其他问题,例如,控制器的客户端验证、可适用工作流程和动态策略模式在什么地方?但是,目前这太容易成为吹毛求疵的问题,有些问题是无关紧要的,或者说应该对 1.0 发行版提这些问题。随着 Struts 小组的不断努力,到您阅读本文时 Struts 说不定已经有了这些功能,或者它很快就会具有这些功能。
参考资源)。

  • 复杂性
    在将问题分为几个部分的同时也引入了复杂性。毫无疑问,要理解 Struts 必须接受一定的培训。随着变化的不断加入,这有时会令人很沮丧。欢迎访问本网站。

  • 在何处...
    我还能指出其他问题,例如,控制器的客户端验证、可适用工作流程和动态策略模式在什么地方?但是,目前这太容易成为吹毛求疵的问题,有些问题是无关紧要的,或者说应该对 1.0 发行版提这些问题。随着 Struts 小组的不断努力,到您阅读本文时 Struts 说不定已经有了这些功能,或者它很快就会具有这些功能。
  • Struts 的前景
    在这个软件开发的新时代,一切都变得很快。在不到 5 年的时间内,我已经目睹了从 cgi/perl 到 ISAPI/NSAPI、再到使用 VB 的 ASP、一直到现在的 Java 和 J2EE 的变迁。Sun 正在尽力将新的变化反映到 JSP/servlet 体系结构中,正如他们对 Java 语言和 API 所作的更改一样。您可以从 Sun 的网站获得新的 JSP 1.2 和 Servlet 2.3 规范的草案。此外,一个标准 JSP 标记库即将出现;有关这些规范和标记库的链接,请参阅参考资源。

    最后的注释
    Struts 使用标记和 MVC 解决了某些重大问题。这个方法有助于提高代码的可重用性和灵活性。通过将问题划分为更小的组件,当技术空间或问题空间中出现变化时,您就有更多的机会重用代码。此外,Struts 使网页设计人员和 Java 开发人员能将精力集中于自己最擅长的方面。但是,在强健性增强的同时,也意味着复杂性的增加。Struts 比简单的单个 JSP 网页要复杂得多,但对于更大的系统而言,Struts 实际上有助于管理复杂性。另外,我并不想编写自己的 MVC 实现,而只想了解一个这样的实现。不管您是否会使用 Struts,回顾这个 Struts 框架(对不起,应该是库)都会使您对 JSP 文件和 servlet 的特性、以及如何将它们组合起来用于您的下一个 Web 项目有更好的了解。正像翼间支柱是机翼结构中不可缺少的一部分一样,Strut 也可能成为您下一个 Web 项目的不可缺少的一部分。

    参考资源

    • 您可以下载本文中所用的代码:business.zip 用来构建 business.jar 文件,joinStruts.zip 用来构建 joinStruts.war 文件。

    • 有关 Struts 的文档、安装说明和下载,请参阅 Struts 的主页。

    • 您可以从 Jakarta 项目下载最近的 Struts 实现。

    • 如想入门,我建议您下载 JSP servlet 的参考实现 Tomcat 3.2。如果您使用 Tomcat 以外的容器,请访问 Struts 用户邮件列表,以获得在您特定的环境中安装 Struts 的帮助。

    • 要获得文档或安装指南中未包含的信息,请参阅 Struts 用户邮件列表归档文件。

    • Apache 邮件列表归档文件包含与其他资料一起归档的 Struts 信息。

    • 要了解将构建和测试过程组合在一起的详细信息,请参阅 "Incremental development with Ant and JUnit"。

    • 请参阅 Proposed Final Draft for Java Servlet 2.3 and JavaServer Pages 1.2 Specifications.

    • 请参阅 Standard Tag Library for JavaServer Pages,JSR #000052。
    business.zip 用来构建 business.jar 文件, joinStruts.zip 用来构建 joinStruts.war 文件。

  • 有关 Struts 的文档、安装说明和下载,请参阅 Struts 的主页。

  • 您可以从 Jakarta 项目下载最近的 Struts 实现。

  • 如想入门,我建议您下载 JSP servlet 的参考实现 Tomcat 3.2。如果您使用 Tomcat 以外的容器,请访问 Struts 用户邮件列表,以获得在您特定的环境中安装 Struts 的帮助。

  • 要获得文档或安装指南中未包含的信息,请参阅 Struts 用户邮件列表归档文件。

  • Apache 邮件列表归档文件包含与其他资料一起归档的 Struts 信息。

  • 要了解将构建和测试过程组合在一起的详细信息,请参阅 "Incremental development with Ant and JUnit"。

  • 请参阅 Proposed Final Draft for Java Servlet 2.3 and JavaServer Pages 1.2 Specifications.

  • 请参阅 Standard Tag Library for JavaServer Pages,JSR #000052。
  • 作者简介
    Malcolm G. Davis 住在阿拉巴马州伯明翰市,他在自己的咨询公司当总裁。他自称是一名 Java 传道者。他在宣传 Java 的优点的闲暇之余,他会去长跑,或者与自己的孩子一起玩。可以通过 malcolm@nuearth.com 与 Malcolm 联系。

    malcolm@nuearth.com 与 Malcolm 联系。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.luyixian.cn/news_show_905352.aspx

如若内容造成侵权/违法违规/事实不符,请联系dt猫网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

哪些权重7权重8的网站怎么做的!我的站就是这么做的!轻量级泛目录无需数据库适合所有网站所有cms只需要放在根目录即可

如题&#xff1a;哪些权重7 权重8的站都是怎么做上来的&#xff01; 疫情期间大家都不好赚钱&#xff0c;出不了门&#xff0c;我偶尔让朋友看了一下我的站&#xff01; 他也想做网站了&#xff0c;然后问我要插件&#xff01;先上图 https://www.bilibili.com/video/BV1me41…

SEO,关键词扩充,文章自动生成,相关说明教程

比如说你的文章标题是&#xff1a;怎么写好一篇作文&#xff08;好作文怎么写&#xff09; ​ 编辑切换为居中 添加图片注释&#xff0c;不超过 140 字&#xff08;可选&#xff09; 通过插件获取标题然后 获得更多相关词 ​ 编辑切换为居中 添加图片注释&#xff0c;不超…

趋势迷-趋势网-网站源码打包下载源码开源无需数据库qushimi.com

趋势迷 趋势网 网站源码是一个php开发的程序&#xff01; 全站无需数据库&#xff01; 添加一点关键词即可甩手&#xff01; 现 整站打包 开源无加密 无授权&#xff01; 无论是 学习 还是运营 养站 都适合&#xff01;&#xff01;&#xff01;&#xff01; 直接服务器 打…

2022个人博客网站搭建笔记

历时11天我的第一个博客网站上线了。作为一个善于“反思”的中国人&#xff0c;当然要给自己来一把复盘和总结了。然后这篇笔记我就鸽了好几天&#xff0c;嘤嘤嘤。 www.ganggangfalcon.xyz 其实说实话&#xff0c;博客网站的搭建难度不大&#xff0c;比较耗时的地方在于网站的…

如何实现“Ipv6网站在线检测工具”

随着IPv6的普及&#xff0c;很多网站已经支持IPv6的访问&#xff0c;我们公司很多客户就要求网站支持IPv6&#xff0c;我们目前是推荐客户使用阿里云的IPv6的产品来实现。但是客户问题就来了&#xff0c;怎么检测网站支持IPv6访问呢&#xff0c;目前我们使用国家IPv6发展检测平…

如何用个人电脑利用ipv6搭建网站

如何用个人电脑利用ipv6搭建网站 需要的工具写在前面的话第一步第二步 需要的工具 一台拥有ipv6地址的个人电脑&#xff08;Windows版&#xff09; 没啦 ^ - ^ 写在前面的话 小白文章&#xff0c;请大佬指正 随着现在ipv6技术的普及&#xff0c;越来越多的电脑&#xff0c;…

本地做个注册会员的网站(非网络

做网站,先得有工具.我用的是Adobe DreamWeaver CS6. 做数据库表格,也得有工具,我用的是Microsoft Access 2007 做上面两者的连接,用的是DSP 还有自己的服务器IIS~~好吧不是6.0..系统自带的OJZ 第一步:建表 打开Access. 然后就按下图的1,2,3来操作 创建完之后,在菜单找到【…

2018年最新 Java实战开发今日头条资讯网站

课程目录 第1节 开发工具和Java语言介绍 主要介绍项目所需要的开发工具&#xff0c;并且会简单回顾这个项目所用到的语言-java&#xff0c;语法基础&#xff0c;控制流&#xff0c;数据结构&#xff0c;面向对象&#xff0c;异常&#xff0c;随机数等。 第2节 Spring入门和模板…

游戏网站博客

一、项目简介 1.1项目博客地址 https://blog.csdn.net/qq_16255439/article/details/103550399 1.2 项目完成的功能与特色 基于python语言&#xff0c;通过使用flask框架&#xff0c;pymysql&#xff0c;JS&#xff0c;CSS&#xff0c;html5&#xff0c;定义蓝图&#xff0c;蓝…

利用拼接字符串的方式来写一个简化版的后台网站(推荐初学者进)

今天用一般处理程序ashx做一个如下的效果&#xff0c; 主要是为了实现功能&#xff0c;界面丑就丑把。 先是显示界面DomeHandler.ashx public void ProcessRequest (HttpContext context) {context.Response.ContentType "text/html";StringBuilder sb new Strin…

asp打开网站原理

因为对如何请求网站的有疑问&#xff0c;所以查了一些资料并画了一张图进行分析。 在解释这张图之前&#xff0c;先了解一下以下相关知识&#xff1a; &#xff08;1&#xff09;内核模式&#xff1a;也可以叫管理员模式&#xff0c;主要运行底层的驱动和监视程序等&#xff0…

python 破解网站反爬虫的两种简单方法

最近在学爬虫时发现许多网站都有自己的反爬虫机制&#xff0c;这让我们没法直接对想要的数据进行爬取&#xff0c;于是了解这种反爬虫机制就会帮助我们找到解决方法。 常见的反爬虫机制有判别身份和IP限制两种&#xff0c;下面我们将一一来进行介绍。 目录 (一) 判别身份(二) …

python爬虫——通过API爬取动态网站的数据

加粗样式在我前面的博客中&#xff0c;通过利用python的requests库和BeautifulSoup库对静态网站进行爬取&#xff0c;但如果遇到动态网站怎么办呢&#xff1f;接下来我们试着通过API来对动态网站进行爬取想要的数据。 目录 (一) 动态网站和静态网站的区别与robots.txt(二) 爬取…

Android 自动抓取网站图标实现分享样式的定制

分享是一个app产生用户数据的重要来源&#xff0c;也是app宣传拉新的重要途径&#xff0c;所以对于稍微成熟一点的app分享是必不可少的模块。相信稍微接触过分享的人都清楚&#xff0c;分享到外部app很简单&#xff0c;只要接入微信、微博、QQ等提供好的sdk并按照规定好的参数正…

程序员命名必备网站|变量命名

程序员命名必备网站 话不多说&#xff0c;之间上网址 https://unbug.github.io/codelf/点击访问 方便又快捷啊兄弟们

浅谈web网站架构演变过程

前言 我们以 javaweb为例&#xff0c;来搭建一个简单的电商系统&#xff0c;看看这个系统可以如何一步步演变。 该系统具备的功能&#xff1a; 用户模块&#xff1a;用户注册和管理商品模块&#xff1a;商品展示和管理交易模块&#xff1a;创建交易和管理 阶段一、单机构建网站…

网站行为日志信息统计分析

网站行为日志信息统计分析 一、开发环境二、项目思路三、系统实现(一)、原始数据上传hdfs(二)、数据清洗&#xff08;第一遍&#xff09;(三)、数据清洗&#xff08;第二遍&#xff09;(三)、通过hive对数据分析 四、总结五、完整代码&#xff1a;(一)、pom.xml文件(二)、初始数…

Bliface借区块链定义视频网站3.0

超级解霸 能看懂这个词的多数都已当爹了&#xff0c;那是一个没有还没有在线播放的年代&#xff0c;光盘解码器是电脑的标配&#xff0c;后来逐渐有人在BBS上享视频给大家下载&#xff0c;我们暂且定为第一代 土豆 2005年4月15日土豆网正式上线&#xff0c;在线视频网站如雨后…

IE10 如何保护您的电脑免于恶意网站危害

NSS 实验室不久前发表了一份关于浏览器拦截恶意程序的 评估报告 (英文) 。在这个报告中可以看出最安全的浏览器是 Internet Explorer 10 。 IE10 可以拦截 99.1% 的已知恶意程序。日常生活中在网络上进行的社交活动、购物、工作等皆有可能会将您的个人信息透露给其他人。Micro…

2017年9月3日 实现网站的权限管理

现在各个企业管理网站对登录的账号都要进行权限管理&#xff0c;并且相当重要&#xff0c;每个账号登录进去所能看到的东西大不相同&#xff0c;下面是实现该功能的一个的一种方法。 需求&#xff1a; 权限&#xff1a;权限是使用者操作系统中功能模块的能力&#xff0c;如“角…