剖析Struts1中的五个对象

Struts是一个基于MVC架构的框架(框架实现了某些领域通用完备功能的底层服务),它主要用于开发Web应用程序,帮助我们减少了用MVC开发Web应用的时间,简化了开发过程,使开发更具模块化、灵活性和重用性。可以说Struts把MVC的设计思想发挥到了极致,尤其在Controller层。

在用Struts开发项目直接接触的有这么几个类:ActionMapping、ActionServlet、ActionForm、Action和ActionForward,这五个类各司其职,使上有老下有小的Controller层接近完美。因Struts是面向对象设计,掌握了这几个类,也就基本掌握了Struts的用法,下面按照这几个类的执行流程举例说明他们在Struts中的具体用法。下图为Struts的一个大致流程图:

剖析Struts1中的五个对象_第1张图片

这里不考虑各个对象的创建细节,单从工作的流程分析各个类的作用。

   ActionServlet

剖析Struts1中的五个对象_第2张图片

客户端向服务器(Tomcat)发起请求,通过在web.xml中的配置,请求直接进入ActionServlet,从命名就可以看出这是一个Servlet,此类是不需要我们显式去创建的,Struts框架已对它做了实现。请求继续调用ActionServlet的doGet/doPost方法,实际上这个两个方法共同调用了RequestProcess类的process方法,process方法才是真正的核心,在process方法中,通过调用processPtah方法截取Request中传递过来的URL,然后调用processMapping方法根据截取的URL取得相应的ActionMapping。

在web.xml中的配置代码如下:

[html]  view plain copy
 
  1. <servlet>  
  2.   <servlet-name>action</servlet-name>  
  3.   <servlet-class>org.apache.struts.action.ActionServlet</servlet-class>  
  4.   <init-param>  
  5.     <param-name>config</param-name>  
  6.     <param-value>/WEB-INF/struts-config.xml</param-value>  
  7.   </init-param>  
  8.   <init-param>  
  9.     <param-name>debug</param-name>  
  10.     <param-value>2</param-value>  
  11.   </init-param>  
  12.   <init-param>  
  13.     <param-name>detail</param-name>  
  14.     <param-value>2</param-value>  
  15.   </init-param>  
  16.   <load-on-startup>2</load-on-startup>  
  17. </servlet>  
  18. <servlet-mapping>  
  19.   <servlet-name>action</servlet-name>  
  20.   <url-pattern>*.do</url-pattern>  
  21. </servlet-mapping>  

从上面配置不难看出,ActionServlet在Tomcat启动时,就已经创建好了,只要是以do结尾的请求都会被ActionServlet拦截下来,然后它将不同的请求转发给对应的Action对象,让Action进一步处理客户端的请求。因此ActionServlet肩负着中央控制器角色,是Struts的核心。下面介绍的各个类中,依旧会看到ActionServlet的影子。

  ActionMapping

剖析Struts1中的五个对象_第3张图片

ActionServlet调用processMapping方法,此方法首先会调用moduleConfig.findActionConfig(path),获取对应的ActionMapping对象。Struts框架将核心配置文件struts-config.xml解析并放入了ActionMapping对象里。

等返回ActionMapping对象后,processMapping方法把ActionMapping设置到Request或Session中方便后面其他地方的使用,体现了面向对象封装的好处。struts-config.xml配置信息如下:

[html]  view plain copy
 
  1. <struts-config>  
  2.     <form-beans>  
  3.         <form-bean name="itemForm" type="com.snail.drp.web.forms.ItemActionForm"/>  
  4.     </form-beans>  
  5.       
  6.     <global-exceptions>  
  7.         <exception key="errors.detail" type="com.snail.drp.AppException" path="/error.jsp"/>  
  8.     </global-exceptions>  
  9.        
  10.     <action-mappings>  
  11.         <action path="/item"  
  12.                 type="com.snail.drp.web.actions.ItemAction"  
  13.                 name="itemForm"  
  14.                 scope="request"  
  15.                 parameter="command"  
  16.                 >  
  17.             <forward name="list" path="/WEB-INF/jsp/item_maint.jsp"/>   
  18.             <forward name="show_add" path="/WEB-INF/jsp/item_add.jsp"/>     
  19.             <forward name="item_index" path="item.do" redirect="true"/>  
  20.             <forward name="show_modify" path="/WEB-INF/jsp/item_modify.jsp"/>  
  21.             <forward name="show_detail" path="/WEB-INF/jsp/item_detail.jsp"/>  
  22.             <forward name="show_upload" path="/WEB-INF/jsp/item_upload.jsp"/>  
  23.         </action>  
  24.     </action-mappings>  
  25.     <message-resources parameter="MessageResources"/>  
  26. </struts-config>  

配置信息里看出,其中中包含有与请求对应的ActionForm、Action、ActionForward、错误处理以及国际化等配置信息,这些都可以通过ActionMapping取出来。

 

  ActionForm

剖析Struts1中的五个对象_第4张图片

processMapping方法生命周期结束后,ActionServlet继续调用processForm方法。ProcessForm方法根据ActionMapping中的name名称查找ActionForm,如果配置了ActionForm,那么就到request或session中查找,如果在request或session中存在已经创建的ActionForm,那么将返回。如果不存在那么会根据ActionForm的完成路径采用反射进行创建,再将创建好的ActionForm放到request或session中,方便后面的读取。

创建好ActionForm对象后,接下来就要该给ActionForm中赋值了,此时,ActionServlet会调用processPopulate方法,首先执行ActionForm中的reset方法进行重置,然后得到表单中所有输入域的name名称,再调用request.getParameterValues(),根据name名称得到相应的值,最后将表单中的数据全部放到一个map中,map的key为表单输入域的名称,map的value为表单输入域的值(字符串数组),接下来调用一个第三方组件BeanUtils,将Map中的值,根据ActionForm中的类型先转换好,再调用ActionForm中的setter方法设置到ActionForm上。

唯一要我们做的是创建一个类,然后让它继承ActionForm,根据表单输入域的name提供get和set方法,并在Struts-Config.xml中配置(参见上一篇)。ActionForm代码如下:

[java]  view plain copy
 
  1. package com.snail.drp.web.forms;  
  2.   
  3. import org.apache.struts.action.ActionForm;  
  4. import org.apache.struts.upload.FormFile;  
  5.   
  6. public class ItemActionForm extends ActionForm {  
  7.   
  8.     private String itemNo;  
  9.       
  10.     private String itemName;  
  11.   
  12.     public String getItemNo() {  
  13.         return itemNo;  
  14.     }  
  15.   
  16.     public void setItemNo(String itemNo) {  
  17.         this.itemNo = itemNo;  
  18.     }  
  19.   
  20.     public String getItemName() {  
  21.         return itemName;  
  22.     }  
  23.   
  24.     public void setItemName(String itemName) {  
  25.         this.itemName = itemName;  
  26.     }  
  27.   
  28. }  

那么一个完整的ActionForm就创建好了,类型转换等工作它也为我们完成了。接下来在Action中就可以任意调用了。其实,ActionForm非常类似于Domain模型中的JavaBean,他和表单中的输入域形成映射关系,将输入域中的name和值通过Set方法放入Map中,其他地方可以通过get方法来获取。

  Action和DispatchAction

剖析Struts1中的五个对象_第5张图片

继续沿着ActionServlet前行,搞定ActionForm后,ActionServlet会调用processActionCreate方法,根据Action的完整类名称到ActionMapping中去查找,如果存在就返回Action对象,否则根据Action类的完整名称采用反射去创建,再将创建好的Action放到ActionMapping中,因此Struts1的Action是单实例的,存在线程安全问题。

然后ActionServlet调用processActionPerform方法,执行用户自定义的Action中的execute方法,将ActionMaping、ActionForm、request、response传递过去,将ActionForward对象返回。

Action是一个Java类,负责调用业务逻辑方法,检测处理异常,校验输入数据和根据逻辑进行转向,它一般按照职责划分,意味着职责的增多,Action类也会相应的增多,为了解决这种类过多的情况,由DispatchAction类代替Action,由它继承Action,把Action众多的类完成的功能放到了方法体中,因此可以在一个DispatchAction中处理多个Action所完成的事情,有效避免了类过多的弊病。代码如下:

[java]  view plain copy
 
  1. package com.snail.drp.web.actions;  
  2.   
  3. import java.io.FileOutputStream;  
  4. import java.util.List;  
  5. import javax.servlet.http.HttpServletRequest;  
  6. import javax.servlet.http.HttpServletResponse;  
  7.   
  8. import org.apache.commons.beanutils.BeanUtils;  
  9. import org.apache.struts.action.ActionForm;  
  10. import org.apache.struts.action.ActionForward;  
  11. import org.apache.struts.action.ActionMapping;  
  12. import org.apache.struts.actions.DispatchAction;  
  13. import org.apache.struts.upload.FormFile;  
  14.   
  15. import com.snail.drp.BeanFactory;  
  16. import com.snail.drp.PageModel;  
  17. import com.snail.drp.domain.Item;  
  18. import com.snail.drp.domain.ItemCategory;  
  19. import com.snail.drp.domain.ItemUnit;  
  20. import com.snail.drp.service.DataDictService;  
  21. import com.snail.drp.service.ItemService;  
  22. import com.snail.drp.web.forms.ItemActionForm;  
  23.   
  24. public class ItemAction extends DispatchAction {  
  25.       
  26.     /** 
  27.      * 查找所有物料信息 
  28.      */  
  29.     @Override  
  30.     protected ActionForward unspecified(ActionMapping mapping, ActionForm form,  
  31.             HttpServletRequest request, HttpServletResponse response)  
  32.             throws Exception {  
  33.         ItemActionForm iaf = (ItemActionForm)form;   
  34.       
  35.         int pageSize = Integer.parseInt(request.getSession().getServletContext().getInitParameter("pageSize"));  
  36.         String queryString = iaf.getClientIdOrName();  
  37.         int pageNo = iaf.getPageNo();  
  38.         ItemService itemService = (ItemService)BeanFactory.getInstance().getBean(ItemService.class);  
  39.         PageModel pageModel =  itemService.findAllItem(queryString, pageNo, pageSize);  
  40.         request.setAttribute("pageModel", pageModel);  
  41.         return mapping.findForward("list");  
  42.     }  
  43.       
  44.     /** 
  45.      * 添加物料 
  46.      * @param mapping 
  47.      * @param form 
  48.      * @param request 
  49.      * @param response 
  50.      * @return 
  51.      * @throws Exception 
  52.      */  
  53.     public ActionForward add(ActionMapping mapping, ActionForm form,  
  54.             HttpServletRequest request, HttpServletResponse response)  
  55.             throws Exception {  
  56.           
  57.         ItemActionForm iaf = (ItemActionForm)form;  
  58.           
  59.         Item item = new Item();  
  60.           
  61.         BeanUtils.copyProperties(item, iaf);  
  62.           
  63.         ItemCategory itemCategory = new ItemCategory();  
  64.         itemCategory.setId(iaf.getCategory());  
  65.         item.setItemCategory(itemCategory);  
  66.           
  67.         ItemUnit itemUnit = new ItemUnit();  
  68.         itemUnit.setId(iaf.getUnit());  
  69.         item.setItemUnit(itemUnit);       
  70.           
  71.         ItemService itemService = (ItemService)BeanFactory.getInstance().getBean(ItemService.class);  
  72.         itemService.addItem(item);  
  73.         return mapping.findForward("item_index");  
  74.     }  
  75. }  

如果将以上代码继承自Action,那么该类必须重写execute方法,执行需要的功能也必须在execute方法中,这样如果不利用多态创建多个Action类,那就得需要很多if...else...语句,这种编码方式是不可取的。

  ActionForward

好了,终于能看到完整的图了。

剖析Struts1中的五个对象_第6张图片

其实从命名就可以看出ActionForward主要负责转向,执行完processActionPerform方法后,返回的就是ActionForward对象,ActionServlet紧接着调用processForwardConfig方法,根据ActionForward完成转向(转发或重定向)。ActionForward主要信息来自于ActionMapping对象,因此需要我们做的是配置好Struts-Config.xml文件即可。

这只是简单的分析了一下应用Struts的流程,要想真正的学习还需研究其源代码来的直接。

你可能感兴趣的:(struts1)