Struts的MVC模式实现机制

 

 

MVC设计模式概述

   MVCModelViewControler的简称。即模型—视图—控制器。最近几年被业界推荐为SUN公司的J2EE平台的设计模式。受到越来越多的WEB开发者的欢迎。MVC是一种设计模式,它强制性的把应用程序的输入、处理和输出分开。MVC中的模型、视图、控制器它们分别担负着不同的任务。
  
视图:视图是用户看到并与之交互的界面。视图向用户显示相关的数据,并接受用户的输入。视图不进行任何业务逻辑处理。
   
模型:模型表示业务数据和业务处理。相当于JavaBean。一个模型能为多个视图提供数据。这提高了应用程序的重用性
  
控制器:当用户单击Web页面中的提交按钮时,控制器接受请求并调用相应的模型去处理请求。然后根据处理的结果调用相应的视图来显示处理的结果。
   MVC
的处理过程:首先控制器接受用户的请求,调用相应的模型来进行业务处理,并返回数据给控制器。控制器调用相应的视图来显示处理的结果。并通过视图呈现给用户。可进行以下类比:凶杀案发生了(用户点击提交按钮)视图--->公安部门进行刑事侦察,找出主凶(控制器)---->交给法院案件审理判刑(模型)---->对主凶执行伏法(视图)
 MVC
的优点

   1:MVC设计模式强制性的把用户的输入、业务逻辑的处理、数据结果的显示分开。提高了Web应用程序的可维护性。比如:改动模型的业务逻辑代码不会影响到视图和控制器。由于MVC三个模块相当于独立,改变 任何一个不会影响其它两个从而实现的程序的松耦合。
   2:
由于多个视图可以共享同一个模型,所以大大提高了Web应用程序的重用性。
   3:MVC
设计模式可以与xml配置文件有机的结合在一起。在改变三个模块间的工作方式,不需要修改源代码,直接修改配置文件就行了。从而提高了Web应用程序的灵活性。

struts的体系结构
1:
用来实现mvc设计模式的框架有很多,比如:webwork,struts等。struts是在j2ee中用来实现mvc模式的主流  框架。后来webworkstruts合并成为struts2.0版本。也即是最高版本。struts的体系结构的实现如图:

   ActionServlet是一个FrontController,是struts的核心控制器。它是一个标准的Servlet,继承自HttpServlet类,该控制器负责拦截所有Http请求,然后把拦截的用户请求根据用户需要,它将request转发给 RequestProcessor来处理, ActionMappingActionConfig的子类,实质上是对struts-config.xml的一个映射,从中可以取得所有的配置信息 ,转发给相应的Action类进行处理。
  RequestProcessor根据提交过来的url,如*.do,从ActionMapping 中得到相应的ActionFornAction。然后将request的参数对应到ActionForm中,进行form验证。如果验证通过则调用 Actionexecute()方法来执行Action,最终返回ActionFoward

   ActionFoward是对mapping中一个foward的包装,对应于一个url

   ActionForm使用了ViewHelper模式,是对HTMLform的一个封装。其中包含有validate方法,用于验证form数据的有 效性。ActionForm是一个符合JavaBean规范的类,所有的属性都应满足getset对应。对于一些复杂的系统,还可以采用 DynaActionForm来构造动态的Form,即通过预制参数来生成Form。这样可以更灵活的扩展程序。

   ActionErrors是对错误信息的包装,一旦在执行action或者form.validate中出现异常,即可产生一个ActionError 最终加入到ActionErrors。在Form验证的过程中,如果有Error发生,则会将页面重新导向至输入页,并提示错误。
   Action
是用于执行业务逻辑的RequsestHandler。这个类通常由用户来提供,该控制器负责接收来自于ActionServlet的请求,并根据该请求调模式的业务逻辑方法处理请求,并将处理的结果返回给jsp页面显示。每个Action都只建立一个instanceAction不是线程安全的,所以 不应该在Action中访问特定资源。一般来说,应改使用 Business Delegate 模式来对Business tier进行访问以解除耦合。

   Struts提供了多种Action供选择使用。普通的Action只能通过调用execute执行一项任务,而DispatchAction可以根据配置参数执行,而不是仅进入execute()函数,这样可以执行多种任务。如insert, update等。LookupDispatchAction可以根据提交表单按钮的名称来执行函数。
 struts组件的MVC实现
     View部分:由jsp页面实现,并且struts提供了丰富的标签库,通过使用struts标签库可以快速实现view部分,当然也可以不用struts标签库,而用普通的jsp页面
     Controler:
有两个控制器。
       A:
核心控制器(ActionSerlvet,只有一个,struts自动创建)
       B:
业务逻辑控制器(Action,有多个,由用户自己创建)。业务逻辑控制器负责处理用户请求,它本身不具备处理能力,而是调用Model来完成处理。

   Model部分:ActionFormJavaBean组成。其中ActionFrom用于封装用户在jsp页面中所填写的数据。所有的用户填写的数据由系统自动封装成ActionFrom对像。该对像被ActionServlet转发给ctionAction根据从ActionFrom中取得用户所填写的数据,交给JavaBean进行处理。ActionFrom实际上也是一个普通的JavaBean。它对应于jsp页面中各个输入数据项 

struts工作流程
1.
客户端填写完jsp页面的数据之后单击提交按钮向服务器发出http请求,客户端所有的请求都被 ActionSerlvet拦截。
2.ActionServlet
根据用户的请求在struts-conifg.xml配置文件中查找与之对应的Action,同时把用户的数据填 充给对应的ActionForm,然后把ActionForm一起分发给Action进行处理。
3.Action
会自动调用excute方法,在该方法里面Action将从ActionForm里面获取用户填写的数据,并调用模型(javaBean)进行相应的处理,然后把处理的结果显示在其它的视图(jsp页面)

struts-config.xml:struts的配置文件.

上面讲到每一个用户的请求是通过ActionSerlvet来处理和转发的那么ActionSerlvet如何决定把用户请求转发给那一个Action呢?这就需要一个配置文件来保存一个struts中所有的Action。而struts-config.xml这个配置文件就是用来实现以上配置的          

配置ActionSerlvet拦截用户请求
 1:ActionSerlvet
是核心控制器,是由struts框架自动产生的,它像过滤器一样会拦截所有用户的请求。但是这个过滤器必须要在web.xml文件中进行配置才能够发挥它的作用。幸运的是这些配置struts已经帮我们自动配置好,但是有必要进行了解.
   <servlet-name>action</servlet-name> --ActionServlet
的名字对应实例对应的类

    <servlet-class>org.apache.struts.action.ActionServlet</servlet-class>
       ........
     <servlet-mapping>
    <servlet-name>action</servlet-name>
    <url-pattern>*.do</url-pattern> --
定义ActionServlet过滤的通配符。必须要以do结尾。当然可以修改
  </servlet-mapping>     

配置ActionForm接受用户输入的数据
  1)ActionForm
实际上就是一个非常简单的javaBean。在它里面有许多的get,set方法,它的目的就是为了封装     从用户输入的数据。因此ActionForm通常与表单页面对应,每个表单对应一个ActionForm。表单中的输入项对应ActionForm的一个属性.
  2)Struts
要求ActionForm必须要继承自org.apache.struts.action.ActionForm基类。

  3)
必须保证ActionForm中的属性名一定要与表单中元素名一致
  4)
所有的ActionForm必须要在Struts-config.xml中进行配置。配置节如下:
      <form-beans>
  <form-bean name="loginActionForm" type="org.chen.struts.loginForm" />
                <form-bean name="shopActionForm"  type="org.chen.struts.loginForm" />
                 ........
                <form-bean name="shopActionForm"  type="org.chen.struts.loginForm" />
 </form-beans>
  5)
一个ActionForm必须至少指定两项。
     name:
该项用于标识ActionFormId,以便于传递给Action进行使用
     type:ActionForm
的实现类
  6)
eclipse中添加一个ActionForm的步骤如下:
      A:
右击--->new--->other--->myeclipse--->webstruts-->struts1.1 form
      B:
在弹出的对话框中

         name
中输入ActionForm的标识id
         superclass
指定ActionForm的父类名。一般选择
org.apache.struts.action.ActionForm
         formtype
指定ActionForm的类名,因为它是一个javaBean。相当于为类取一个名字

         formProperty
选项卡中输入ActionForm的属性。    

Action的创建与配置
  1)Action
是系统的业务控制器,它用于接收从ActionServlet转发过来的请求,并触发execute方法,我们可以    execute方法里面调用模型进行数据处理
  2)struts
要求Action必须继承自org.apache.struts.action.Action基类
  3)
所有的Action必须要在struts-config.xml中进行配置。配置节如下:
        <action   name="loginActionForm"
   path="/loginAction" scope="request"
   type="org.chen.struts.loginAction">
   <forward name="scuess" path="/scuess.jsp" />
   <forward name="error" path="/error.jsp" />
        </action>
    
说明
:
A)name:
并不是Action本身的名字,而是与之关联的ActionForm的名称

B)path:
非常重要的属性。ActionSerlvet将用户的请求转发与之同名的Action.同名的意思是将请求 ".do"后缀去掉。匹配Actionpath属性值
C)struts-config.xml
中只要遇到path属性都要加"/"表示相当于WebRoot根目录而言

D)type:Action
的类名
E)forward:
Action的转发映射到实际的jsp页面,在实际编程时应该用逻辑名进行转发
F)
一个Action可以有ActionForm,也可以没有ActionForm。如果是post提交的一般会有ActionForm。比如:<form id="frm1" action="loginAction.do">,如果是通过get提交则不会有ActionForm,但是Action总是会有的,因为要处理业务逻辑。比如:
<a href="shopAddAction.do">
G)
关于超链接与Action路径的问题
(******************)
     [1]
对于表单中的Action而言,不存在路径问题,只要Actionclasses下都可以定位比如: 1.jsp页面在WebRoot下面,但Actionht.netBook.Action下面。在表单中提交时

<html:form action="insertTalkAction.do">
可以直接指定Action名就可以了。或者2.jsp页面在WebRoot下面的page下面,都可以正确定位。
       [2]
但对于超链接而言,则不一样。如果 超链接所在的页面1.jspWebRoot下面,    

则可以<a href="insertTalkAction.do>进入</a>直接指定Action名, 但是如果超链接所在的页面1.jspWebRoot下面的page下面的话,则通过"/"表示,项目目录 <a href="/WebRoot/insertTalkAction.do>struts回到WebRoot下面再去找。
      [3]
如果是servlet的话,不管是对于表单还是对于超链接,都必须要找到web.xml

件在它下面有。取出"servlet/loginSer"就行了,注意不是"/servlet/loginSer",即不要加"/"。如果超链接所在的页面就在WebRoot的下面,则可以直接使用<a href="servlet/loginSer"> 如果在超链接所在的页面在WebRoot下面的Page下面,则把则要把路径回退一级<a href="..\servlet\loginSer>struts回到WebRoot下面再去找
                <servlet-mapping>
                     <servlet-name>loginSer</servlet-name>
                     <url-pattern>/servlet/loginSer</url-pattern>
                </servlet-mapping>
     [4]
对于forward中的<forward name="error" path="/error.jsp" />中的path而言"/"表示在WebRoot下面去找,但是对于jsp而言"/"表示项目目录。
         
如果要在WebRoot下面的page文件夹里面的error.jsp则修改成为<forward name="error" path="/page/error.jsp" />

         举例:
          1)
创建一个Action名为loginAction放入ht.OA.Action下面
          2)
创建一个login.jsp页面放入WebRoot根目录下面
          3)
创建一个index.jsperror.jsp放入WebRootpage文件夹下面
          4)
创建一个servlet放入ht.OA.servlets下面
          5)
完成登录页面的struts的测试 

Forward的定义转向配置
         1)
前面说过ActionServlet把请求转发给ActionAction会触发excute方法,在此方法中调用JavaBean进行业务逻辑处理,处理完毕之后,要把处理的结果显示到其它的jsp页面中。而这些经过Action处理之后要进入的jsp页面就是Forward配置节的任务
         2)Forward
struts-config.xml中配置格式如下:
                        <forward name="scuess" path="/scuess.jsp" />
   <forward name="error" path="/error.jsp" /> 
         3)
每一个Forward相当于一个ActionForward对像,所以struts中的
           ActionSerlvet,Action,ActionForwad
构成了struts的核心
       
说明:
         A)name:
用来指定forward逻辑名,以便在编程中来使用
         B)path:
实际要转发到的jsp页面。注意要加"/"       

Actionexcute方法
        1)ActionServlet
接受到用户的"XXXXX.do"请求,通过struts-config.xml配置文件找到与之匹配的Action配置节,通过Actionname属性找到与之对应的ActionForm然后把用户输入的数据填写到该ActionForm中去,最后把请求连同ActionForm转发给Action,然后触发Actionexecute方法。
        2) execute
方法里面有几个参数:
             A)mapping:
                1:
用来获取与Action对应的forward定义转向。

                2:
通过mapping.findForward(name)来进行实际页面的转发
             B)form:
是填充了用户输入数据的ActionForm。在实际编程时,往往需要向子类转换
                LoginActionForm loginActionForm=(LoginActionForm)form,
得到loginActionForm之后可以通过get方法得到用户填写的数据
             C)request:
得到用户的请求。事实上要得到用户输入的数据也可以通过request.getParameter方法,但是那样就失去了struts的作用,所以一般都会有ActionForm去得到用户所输入的数据,但是request.getParameter方法却可以得到get方法提交过来的数据,这是非常重要的。
             D)response:
得到用户的响应
             E)execute
方法会返回一个ActionForward实例,而mapping.findForward(name)正好可以返回这个实例。

 

本文主要摘自:

1.《开源web框架-Struts工作流程》http://hi.baidu.com/pursuit_on/blog/item/f8b3ef957ecb534bd0135e6d.html

2.《Struts天下》

http://blog.programfan.com/blog.asp?blogid=3274&columnid=6785

 

你可能感兴趣的:(struts)