struts1工作原理浅析(一)

一、引入struts

Model 1结构如图1所示:

    Mode11是一个以JSP文件为中心的模式,在这种模式中JSP页面不仅负责表现逻辑,也负责控制逻辑。专业书籍上称之为逻辑耦合在页面中,这种处理方式,对一些规模很小的项目如:一个简单的留言簿,也没什么太大的坏处,实际上,人们开始接触一些对自己来说是新的东西的时候,比如,用JSP访问数据库时,往往喜欢别人能提供一个包含这一切的单个JSP页面,因为这样在一个页面上他就可以把握全局,便于理解。但是,用Model 1模式开发大型时,程序流向由一些互相能够感知的页面决定,当页面很多时要清楚地把握其流向将是很复杂的事情,当您修改一页时可能会影响相关的很多页面,大有牵一发而动全身的感觉,使得程序的修改与维护变得异常困难;还有一个问题就是程序逻辑开发与页面设计纠缠在一起,既不便于分工合作也不利于代码的重用,这样的程序其健壮性和可伸缩性都不好。                    

Model 2引入了"控制器"这个概念,控制器一般由Servlet来担任,客户端的请求不再直接送给一个处理业务逻辑的JSP页面,而是送给这个控制器,再由控制器根据具体的请求调用不同的事务逻辑,并将处理结果返回到合适的页面。因此,这个servlet控制器为应用程序提供了一个进行前-后端处理的中枢。一方面为输入数据的验证、身份认证、日志及实现国际化编程提供了一个合适的切入点;另一方面也提供了将业务逻辑从JSP文件剥离的可能。业务逻辑从JSP页面分离后,JSP文件蜕变成一个单纯完成显示任务的东西,这就是常说的View。而独立出来的事务逻辑变成人们常说的Model,再加上控制器Control本身,就构成了MVC模式。实践证明,MVC模式为大型程序的开发及维护提供了巨大的便利。

二、struts工作原理

来自客户的所有需要通过框架的请求,统一由ActionServlet接收(ActionServlet Struts已经为我们写好了,只要您应用没有什么特别的要求,它基本上都能满足您的要求),根据接收的请求参数和Struts配置(struts-config.XML)中ActionMapping,将请求送给合适的Action去处理,解决由谁做的问题,它们共同构成Struts的控制器。

Action则是Struts应用中真正干活的组件,它解决的是做什么的问题,它通过调用需要的业务组件(模型)来完成应用的业务,业务组件解决的是如何做的问题,并将执行的结果返回一个代表所需的描绘响应的JSP(或Action)的ActionForward对象给ActionServlet以将响应呈现给客户。

这里要特别说明一下的是:就是Action这个类,它不应该包含过多的业务逻辑,而应该只是简单地收集业务方法所需要的数据并传递给业务对象。实际上,它的主要职责是:

校验前提条件或者声明

调用需要的业务逻辑方法

检测或处理其他错误

路由控制到相关视图

三、struts运行机制

Struts的工作流程:

在web应用启动时就会加载,初始化ActionServlet,ActionServlet从struts-config.xml文件中读取配置信息,把它们存放到各种配置对象中,当ActionServlet接收到一个客户请求时,将执行如下流程.

    (1)检索和用户请求匹配的ActionMapping实例,如果不存在就返回请求路径无效信息;

(2)如果ActionForm实例不存在,就创建一个ActionForm对象,把客户提交的表单数据保存到ActionForm对象中;

(3)根据配置信息决定是否需要表单验证.如果需要验证,就调用ActionForm的validate()方法;

(4)如果ActionForm的validate()方法返回null或返回一个不包含ActionMessage的ActuibErrors对象, 就表示表单验证成功;

(5)ActionServlet根据ActionMapping所包含的映射信息决定将请求转发给哪个Action,如果相应的Action实例不存在,就先创建这个实例,然后调用Action的execute()方法;

(6)Action的execute()方法返回一个ActionForward对象,ActionServlet在把客户请求转发给 ActionForward对象指向的JSP组件;

(7)ActionForward对象指向JSP组件生成动态网页,返回给客户;

四、Struts1的安装与基本配置

我们主要针对Struts1.1版本进行讲解,这里假定读者已经配置好Java运行环境和相应的Web容器,本文例子所使用的是j2sdk和Tomcat4.1.27。下面,将采用类似于step by step的方式介绍其基础部分。

安装Struts到http://jakarta.apache.org/ 下载Struts的安装文件,本文例子使用的是1.1版。接下来您要进行如下几个步骤来完成安装:

1、解压下载的安装文件到您的本地硬盘

2、生成一个新的Web应用,假设我们生成的应用程序的根目录在/Webapps/mystruts目录。在server.xml文件中为该应用新建一个别名如/mystruts

3、从第1步解压的文件中拷贝下列jar文件到/Webapps/mystruts/WEB-INF/lib目录,主要文件有如下一些。

struts.jar commons-beanutils.jar

commons-collections.jar

commons-dbcp.jar

commons-digester.jar

commons-logging.jar

commons-pool.jar

commons-services.jar

commons-validator.jar

4、创建一个web.xml文件,这是一个基于servlet的Web应用程序都需要的部署描述文件,一个Struts Web应用,在本质上也是一个基于servlet的Web应用,它也不能例外。
    Struts有两个组件要在该文件中进行配置,它们是:ActionServlet和标签库。下面是一个配置清单

<?xml version="1.0"encoding="UTF-8"?>

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

<servlet>

<servlet-name>action</servlet-name>

<servlet-class>

org.apache.struts.action.ActionServlet

</servlet-class>

<init-param>

<param-name>config</param-name>

<param-value>/WEB-INF/struts-config.xml</param-value>

</init-param>

<init-param>

<param-name>debug</param-name>

<param-value>2</param-value>

</init-param>

<load-on-startup>2</load-on-startup>

</servlet>

<servlet-mapping>

<servlet-name>action</servlet-name>

<url-pattern>*.do</url-pattern>

</servlet-mapping>

<taglib>

<taglib-uri>/WEB-INF/struts-bean.tld</taglib-uri><taglib-location>/WEB-INF/struts-bean.tld</taglib-location>

</taglib>

<taglib>

<taglib-uri>/WEB-INF/struts-html.tld</taglib-uri><taglib-location>/WEB-INF/struts-html.tld</taglib-location>

</taglib>

<taglib>

<taglib-uri>/WEB-INF/struts-logic.tld</taglib-uri><taglib-location>/WEB-INF/struts-logic.tld</taglib-location>

</taglib>

</web-app>

上面我们在web.xml中完成了对servlet和标签库的基本配置,而更多的框架组件要在struts-config.xml中进行配置:

5、创建一个基本的struts-config.xml文件,并把它放在/Webapps/mystruts/WEB-INF/目录中,该文件是基于Struts应用程序的配置描述文件,它将MVC结构中的各组件结合在一起,开发的过程中会不断对它进行充实和更改。在Struts1.0时,一个应用只能有一个这样的文件,给分工开发带来了一些不便,在Struts1.1时,可以有多个这样的文件,将上述缺点克服了。需在该文件中配置的组件有:data-sources

global-execptions form-beansglobal-forwards action-mappings controller message-resources plug-in

配置清单如下:

<?xml version="1.0"encoding="UTF-8"?>

<!DOCTYPE struts-config PUBLIC"-//Apache Software Foundation//DTD Struts Configuration1.1//EN""http://jakarta.apache.org/struts/dtds/struts-config_1_1.dtd">

<struts-config>

 <message-resourcesparameter="ApplicationResources" />

</struts-config>

因为Struts是建立在MVC设计模式上的框架,你可以遵从标准的开发步骤来开发你的Struts Web应用程序,这些步骤大致可以描述如下:
1、定义并生成所有代表应用程序的用户接口的Views,同时生成这些Views所用到的所有ActionForms并将它们添加到struts-config.xml文件中。
2、在ApplicationResource.properties文件中添加必要的MessageResources项目
3、生成应用程序的控制器。
4、在struts-config.xml文件中定义Views与 Controller的关系。
5、生成应用程序所需要的model组件
6、编译、运行你的应用程序.

五、ActionServlet

我们首先来了解 MVC 中的控制器。在 Struts 1.1 中缺省采用 ActionServlet 类来充当控制器。当然如果 ActionServlet 不能满足你的需求,你也可以通过继承它来实现自己的类。这可以在 /WEB-INF/web.xml 中来具体指定。

要掌握 ActionServlet,就必须了解它所扮演的角色。首先,ActionServlet表示 MVC 结构中的控制器部分,它需要完成控制器所需的前端控制及转发请求等职责。其次,ActionServlet 被实现为一个专门处理 HTTP 请求的 Servlet,它同时具有 servlet 的特点。在 Struts 1.1 中它主要完成以下功能:

接收客户端请求

根据客户端的 URI 将请求映射到一个相应的 Action 类

从请求中获取数据填充 Form Bean(如果需要)

调用 Action 类的 execute() 方法获取数据或者执行业务逻辑

选择正确的视图响应客户

此外,ActionServlet 还负责初始化和清除应用配置信息的任务。ActionServlet的初始化工作在 init 方法中完成,它可以分为两个部分:初始化 ActionServlet 自身的一些信息以及每个模块的配置信息。前者主要通过initInternal、initOther 和initServlet 三个方法来完成。

我们可以在 /WEB-INF/web.xml 中指定具体的控制器以及初始参数,由于版本的变化以及Struts 1.1 中模块概念的引进,一些初始参数被废弃或者移入到/WEB-INF/struts-config.xml 中定义。下面列出所有被废弃的参数,相应地在web.xml 文件中也不鼓励再使用。

application

bufferSize

content

debug

factory

formBean

forward

locale

mapping

maxFileSize

multipartClass

nocache

null

tempDir

ActionServlet根据不同的模块来初始化ModuleConfig 类,并在其中以 XXXconfig 集合的方式保存该模块的各种配置信息,比如 ActionConfig,FormBeanConfig 等。

初始化工作完成之后,ActionServlet 准备接收客户请求。针对每个请求,方法process(HttpServletRequest request, HttpServletResponse response) 将被调用。该方法指定具体的模块,然后调用该模块的 RequestProcessor 的 process 方法。

 protected void process(HttpServletRequest request,

     HttpServletResponse response)

                  throws IOException, ServletException {

          RequestUtils.selectModule(request, getServletContext());

          getRequestProcessor(getModuleConfig(request)).process(request, response);

 }

RequestProcessor 包含了 Struts 控制器的所有处理逻辑,它调用不同的processXXX 方法来完成不同的处理。下表列出其中几个主要的方法:

方法

功能

processPath

获取客户端的请求路径

processMapping

利用路径来获得相应的 ActionMapping

processActionForm

初始化 ActionForm(如果需要)并存入正确的 scope 中

processActionCreate

初始化 Action

processActionPerform

调用 Action 的 execute 方法

processForwardConfig

处理 Action 返回的 ActionForward

六、ActionForm

对于 ActionForm 你可以从以下几个方面来理解它:

1、ActionForm 表示 HTTP 窗体中的数据,可以将其看作是模型和视图的中介,它负责保存视图中的数据供模型或者视图使用。Struts 1.1 文档中把它比作 HTTP 和 Action 之间的防火墙,这体现了 ActionForm 具有的过滤保护的作用,只有通过 ActionForm 验证的数据才能够发送到 Action 处理。

2、ActionForm 是与一个或多个 ActionConfig 关联的 JavaBean,在相应的 action 的 execute 方法被调用之前,ActionForm 会自动利用请求参数来填充自己(初始化属性)。

3、ActionForm 是一个抽象类,你必须通过继承来实现自己的类。

ActionForm 首先利用属性的 getter 和 setter 方法来实现初始化,初始化完毕后,ActionForm 的 validate 方法被调用,你可以在其中来检查请求参数的正确性和有效性,并且可以将错误信息以 ActionErrors 的形式返回到输入窗体。否则,ActionForm 将被作为参数传给 action 的 execute 方法以供使用。

ActionFormbean 的生命周期可以设置为session(缺省)和 request,当设置为session 时,记得在 reset 方法中将所有的属性重新设置为初始值。

由于 ActionForm 对应于 HTTP 窗体,所以随着页面的增多,你的 ActionForm 将会急速增加。而且可能同一类型页面字段将会在不同的ActionForm 中出现,并且在每个 ActionForm 中都存在相同的验证代码。为了解决这个问题,你可以为整个应用实现一个 ActionForm 或者至少一个模块对应于一个 ActionForm。

但是,聚合的代价就是复用性很差,而且难维护。针对这个问题,在 Struts 1.1 中提出了 DynaActionForm 的概念。

七、Action

我们通过继承 Action 类来实现具体的执行类。具体 Action 类的功能一般都在 execute(以前是 perform 方法)方法中完成,其中主要涉及到以下几个方面:

辅助 ActionForm 进行一些表单数据的检查。

执行必要的业务逻辑,比如存取数据库,调用实体 bean 等。

更新服务器端的 bean 数据,后续对象中可能会用到这些数据,比如在 JSP 中利用 bean:write 来获得这些数据。

根据处理结果决定程序的去处,并以 ActionForward 对象的形式返回给 ActionServlet。

提示:由于在 Action 和 ActionForm 中都可以实现验证方法,那么如何来安排它们之间的分工呢?一般来说,我们秉着 MVC 分离的原则,也就是视图级的验证工作放在 ActionForm 来完成,比如输入不能为空,email 格式是否正确,利用 ValidatorForm 可以很轻松地完成这些工作。而与具体业务相关的验证则放入 Action 中,这样就可以获得最大 ActionForm 重用性的可能。

前面我们提到过,我们主张将业务逻辑执行分离到单独的 JavaBean 中,而 Action 只负责错误处理和流程控制。而且考虑到重用性的原因,在执行业务逻辑的 JavaBean 中不要引用任何与 Web 应用相关的对象,比如 HttpServletRequest,HttpServletResponse 等对象,而应该将其转化为普通的 Java 对象。关于这一点,可以参考 Petstore 中 WAF 框架的实现思路。

此外,你可能还注意到 execute 与 perform 的一个区别:execute 方法简单地掷出 Exception 异常,而 perform 方法则掷出 ServletException 和 IOException 异常。这不是说 Struts 1.1 在异常处理功能方面弱化了,而是为了配合 Struts 1.1 中一个很好的功能 -- 宣称式异常处理机制。

八、宣称式异常处理

和EJB 中的宣称式事务处理概念类似,宣称式异常处理其实就是可配置的异常处理,你可以在配置文件中指定由谁来处理Action 类中掷出的某种异常。你可以按照以下步骤来完成该功能:

1、实现org.apache.struts.action.ExceptionHandler 的子类,覆盖execute 方法,在该方法中处理异常并且返回一个 ActionForward 对象

2、在配置文件中配置异常处理对象,你可以配置一个全局的处理类或者单独为每个Action 配置处理类

下表就定义了一个全局的处理类 CustomizedExceptionHandler,它被用来处理所有的异常。

 <global-exceptions>

 <exception

          handler="com.yourcorp.CustomizedExceptionHandler"

          key="global.error.message"

          path="/error.jsp"

          scope="request"

          type="java.lang.Exception"/>

 </global-exceptions>

其中具体的参数含义,可以参考 ExceptionHandler.java 源文件。

九、taglib

讲完了模型和控制器,接下来我们要涉及的是视图。视图的角色主要是由 JSP 来完成,从 JSP 的规范中可以看出,在视图层可以"折腾"的技术不是很多,主要的就是自定义标记库的应用。Struts 1.1 在原有的四个标记库的基础上新增了两个标记库 --Tiles 和 Nested。

其中 Tiles 除了替代 Template 的基本模板功能外,还增加了布局定义、虚拟页面定义和动态页面生成等功能。Tiles 强大的模板功能能够使页面获得最大的重用性和灵活性,此外可以结合 Tiles 配置文件中的页面定义和 Action 的转发逻辑,即你可以将一个 Action 转发到一个在 Tiles 配置文件中定义的虚拟页面,从而减少页面的数量。比如,下表中的 Action定义了一个转发路径,它的终点是 tile.userMain,而后者是你在 Tiles 配置文件中定义的一个页面。

 <!-- ========== Action Mapping Definitions ============================== -->

 <action-mappings>

 <!-- Action mapping for profile form -->

          <action path="/login"

                  type="com.ncu.test.LoginAction"

                  name="loginForm"

                  scope="request"

                  input="tile.userLogin"

                  validate="true">

                  <forward name="success" path="tile.userMain"/>

          </action>

 </action-mappings>

Tiles 配置文件:tiles-defs.xml

 <!DOCTYPE tiles-definitions PUBLIC

 "-//Apache Software Foundation//DTD Tiles Configuration//EN"

 "http://jakarta.apache.org/struts/dtds/tiles-config.dtd">

 <tiles-definitions>

 <!-- =======================================================  -->

 <!-- Master definitions                                       -->

 <!-- =======================================================  -->

 <!-- Page layout used as root for all pages. -->

 <definition name="rootLayout" path="/tiles-layouts/rootLayout.jsp">

     <put name="titleString" value="CHANGE-ME"/>

     <put name="topMenu" value="/tiles-components/topMenu.jsp"/>

     <put name="leftMenu" value="/tiles-components/panel1.jsp"/>

     <put name="body" value="CHANGE-ME"/>

     <put name="footer" value="/tiles-components/footer.jsp"/>

 </definition>

 <!-- =======================================================  -->

 <!-- Page definitions                      -->

 <!-- =======================================================  -->

 <!-- User Login page -->

 <definition name="tile.userLogin" extends="rootLayout">

     <put name="titleString" value="User Login"/>

     <put name="body" value="/src/userLogin.jsp"/>

 </definition>

 <!-- User Main page -->

 <definition name="tile.userMain" extends="rootLayout">

     <put name="titleString" value="User Main"/>

     <put name="body" value="/src/userMain.jsp"/>

 </definition>

 </tiles-definitions>

而 Nested 标记库的作用是让以上这些基本标记库能够嵌套使用,发挥更大的作用。

十、Commons Logging接口

所谓的 Commons Logging 接口,是指将日志功能的使用与日志具体实现分开,通过配置文件来指定具体使用的日志实现。这样你就可以在 Struts 1.1 中通过统一的接口来使用日志功能,而不去管具体是利用的哪种日志实现,有点于类似 JDBC 的功能。Struts 1.1 中支持的日志实现包括:Log4J,JDK Logging API, LogKit,NoOpLog 和SimpleLog。

你可以按照如下的方式来使用Commons Logging 接口(可以参照 Struts 源文中的许多类实现):

 package com.foo;

 // ...

 import org.apache.commons.logging.Log;

 import org.apache.commons.logging.LogFactory;

 //...

          public class Foo {

          // ...

          private static Log log = LogFactory.getLog(Foo.class);

          // ...

          public void setBar(Bar bar) {

                  if (log.isTraceEnabled()) {

                           log.trace("Setting bar to " + bar);

                  }

          this.bar = bar;

          }

 // ...

 }

而开启日志功能最简单的办法就是在 WEB-INF/classes 目录下添加以下两个文件:commons-logging.properties文件:

# Note: The Tiles framework now uses the commons-logging package to output different

information or debug statements.

Please refer to this package documentation to enable it. The simplest way to enable

logging is to create two files in

WEB-INF/classes:

# commons-logging.properties

# org.apache.commons.logging.Log=org.apache.commons.logging.impl.SimpleLog

# simplelog.properties

# # Logging detail level,

# # Must be one of ("trace", "debug", "info", "warn", "error", or "fatal").

#org.apache.commons.logging.simplelog.defaultlog=trace

org.apache.commons.logging.Log=org.apache.commons.logging.impl.SimpleLog

simplelog.properties文件:

 # Logging detail level,

 # Must be one of ("trace", "debug", "info", "warn", "error", or "fatal").

 org.apache.commons.logging.simplelog.defaultlog=fatal

这里我们采用的日志实现是 SimpleLog,你可以在simplelog.properties 文件指定日志明细的级别:trace,debug,info,warn,error 和 fatal,从trace 到 fatal 错误级别越来越高,同时输出的日志信息也越来越少。而这些级别是和 org.apache.commons.logging.log 接口中的方法一一对应的。这些级别是向后包含的,也就是前面的级别包含后面级别的信息。

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值