`
fatedgar
  • 浏览: 131653 次
  • 性别: Icon_minigender_1
  • 来自: 安徽
社区版块
存档分类
最新评论

struts中的RequestProcessor

阅读更多
   在Struts中,擔任 MVC / Model 2 中Controller角色的是ActionServlet,所有的請求都必須先通過它,然而在Struts 1.1後,有關於請求的處理大部份已交由RequestProcessor,當ActionServlet收到GET或POST的請求,其doGet() 或doPost()會呼叫process()方法來處理請求:
protected void process(HttpServletRequest request,
                        HttpServletResponse response)
                   throws IOException, ServletException {
    RequestUtils.selectApplication(request,
                                   getServletContext());
    getApplicationConfig(
         request).getProcessor().process(request, response);
}


RequestUtils是個工具類,ActionServlet呼叫其selectApplication()方法,藉由 request.getServletPath()來取得請求路徑以選擇應用程式模塊來處理請求,之後從ApplicationConfig物件取得 RequestProcessor物件,將使用者的請求委託它來進行處理。

通常是將ActionServlet當作黑盒子,您只要使用它,然而您也可以繼承ActionServlet來定義自己的控制器,但由於在Struts 1.1中大部份的請求已經委託RequestProcessor來處理,繼承ActionServlet來定義自己的控制器請求處理意義已經不大,通常的目的是重新定義ActionServlet的init()方法,增加自己的初始化動作:
public class CustomActionServlet extends ActionServlet {
    public void init() throws ServletException {
        super.init();

        // 增加自己的初始化動作
        ....
    }
}


預設的RequestProcessor物件是org.apache.struts.action.RequestProcessor,您可以藉由觀看 process()方法的原始碼來瞭解它作了哪些事情:
public void process(HttpServletRequest request,
                     HttpServletResponse response)
                  throws IOException, ServletException {
    // 處理 contentType 為 multipart/form-data 的 POST 請求
    request = processMultipart(request);

    // 取得 URI 路徑
    String path = processPath(request, response);
    if(path == null)
        return;
    .....

    // 確定客戶端的位置
    // 是否要將一個Locale物件儲存在 session 中
    // 配合 <controller> 的 locale 屬性使用
    processLocale(request, response);

    // 確定contentType,預設是 text/html
    processContent(request, response);

    // 判斷<controller>屬性nocache是否被設定
    // 若是,在 response 中加入防止快取的header
    processNoCache(request, response);

    // 前置處理,預設返回 true
    //子類可以重新定義它以決定要不要繼續處理
    if(!processPreProcess(request, response)) {
        return;
    }

    // 從URI路徑確定ActionMapping
    ActionMapping mapping =
                   processMapping(request, response, path);
    if(mapping == null) {
        return;
    }

    ....

    // 處理ActionForm
    // 如果沒有就新增一個,之後一直使用它
    ActionForm form =
            processActionForm(request, response, mapping);
   
    // 將表單的欄位值填入ActionForm
    processPopulate(request, response, form, mapping);

    // 判斷是否執行ActionForm的validate()方法
    if(!processValidate(request, response, form, mapping)) {
        return;
    }

    // 判斷 <action> 標籤的 forward 或 include 標籤
    // 是不是被設定,這兩個標籤在設定一個路徑
    // 其與 type 屬性是互斥的,當設定其中一個屬性時
    // 調用 RequestDispatcher 的 forward() 或 include()
    // 其作用與設定ForwardAction或IncludeAction相同
    // 直接繞送而不再使用Action物件進行接下來的處理
    if(!processForward(request, response, mapping)) {
        return;
    }
    if(processInclude(request, response, mapping)) {
        return;
    }

    // 處理Action,如果沒有就生成一個,之後一直使用它
    Action action =
            processActionCreate(request, response, mapping);
    if(action == null) {
        return;
    }

    // 呼叫Action的execute()或perform()方法
    // 並返回ActionForward
    ActionForward forward =
                    processActionPerform(request,
                                         response,
                                         action,
                                         for, mapping);

    // 處理ActionForward
    processActionForward(request, response, forward);
}
 


您可以繼承RequestProcessor,並改寫其中的processXXXXX()方法來自定義請求的處理方式,如果您要使用自己的 RequestProcessor,要在struts-config.xml中使用<controller>標籤來定義,例如:

    struts-config.xml

...
 <controller 
    contentType="text/html;charset=Big5" 
    locale="true" 
    nocache="true"
    processorClass="caterpillar.CustomRequestProcessor"/> 
...


在Struts 1.1後,新增了<controller>標籤,它可以用於指定ActionServlet的一些參數,在Struts 1.1之前,這些參數是在<init-params>中加以指定,使用<controller>標籤,應用程式中不同的模組也可以指定各自的參數給ActionServlet。
Struts框架只允许应用中存在一个ActionServlet类,但是可以存在多个客户化的RequestProcessor类,每个子应用模块都可以有单独的RequestProcessor类,





ActionServlet主要负责初始化,以及介绍请求并找到合适的RequestRrocessor,之后真正干活的是RequestProecssor和Action.
上回说到ActionServlet的process方法最终会调用RequestProcessor类的process方法.下面介绍这个方法.
一.RequestProcessor的process方法

public void process(HttpServletRequest request,   
	                        HttpServletResponse response)   
	        throws IOException, ServletException {   
	        // Wrap multipart requests with a special wrapper   
	        request = processMultipart(request);   
	        // Identify the path component we will use to select a mapping   
	        String path = processPath(request, response);   
	        if (path == null) {   
	            return;   
	        }    
	        if (log.isDebugEnabled()) {   
	            log.debug("Processing a '" + request.getMethod() +   
	                      "' for path '" + path + "'");   
	        }   
	        // Select a Locale for the current user if requested   
	        processLocale(request, response);   
	        // Set the content type and no-caching headers if requested   
	        processContent(request, response);   
	        processNoCache(request, response);   
	        // General purpose preprocessing hook   
	        if (!processPreprocess(request, response)) {   
	            return;   
	        }   
	        this.processCachedMessages(request, response);   
	        // Identify the mapping for this request   
	        ActionMapping mapping = processMapping(request, response, path);   
	        if (mapping == null) {   
	            return;   
	        }   
	        // Check for any role required to perform this action   
	        if (!processRoles(request, response, mapping)) {   
	            return;   
	        }   
	        // Process any ActionForm bean related to this request   
	        ActionForm form = processActionForm(request, response, mapping);   
	        processPopulate(request, response, form, mapping);   
	        // Validate any fields of the ActionForm bean, if applicable   
	        try {   
	            if (!processValidate(request, response, form, mapping)) {   
	                return;   
	            }   
	        } catch (InvalidCancelException e) {   
	            ActionForward forward = processException(request, response, e, form, mapping);   
	            processForwardConfig(request, response, forward);   
	            return;   
	        } catch (IOException e) {   
	            throw e;   
	        } catch (ServletException e) {   
	            throw e;   
	        }   
	               
	        // Process a forward or include specified by this mapping   
	        if (!processForward(request, response, mapping)) {   
	            return;   
	        }   
	        if (!processInclude(request, response, mapping)) {   
	            return;   
	        }   
	        // Create or acquire the Action instance to process this request   
	        Action action = processActionCreate(request, response, mapping);   
	        if (action == null) {   
	            return;   
	        }   
        // Call the Action instance itself   
	        ActionForward forward =   
	            processActionPerform(request, response,   
	                                 action, form, mapping);   
	  
	        // Process the returned ActionForward instance   
	        processForwardConfig(request, response, forward);   
	  
	    }   

1) 调用processMultipart()方法
如果HTTP请求方式为post,并且contentType为”multipart/form-data”开头,标准的HttpServletRequest对象将被重新包装,以方便处理”multipart”类型的HTTP请求.如果请求方式为get,或正congtentType属性不是”mulitipart”,就直接返回原始的HttpServletRequest对象.

2) 调用processPath()方法
获得请求的URI的路径,这一信息可用于选择合适的Struts Action组件.

3) 调用processLocale方法
当ControllerConfig对象的locale属性为true,将读取用户请求中包含的Locale信息,然后把Locale实例保存在session范围内.

4) 调用processContendType(contentType)方法
读取ControllerConfig对象的conttentType属性,然后调用response.setContentType(contentType)方法,设置响应结果的文档类型和字符编码.
processContent()方法如下

	protected void processContent(HttpServletRequest request,   
	                                 HttpServletResponse response) {   
	  
	       String contentType = moduleConfig.getControllerConfig().getContentType();   
	       if (contentType != null) {   
	           response.setContentType(contentType);   
	       }   
	  
	   } 
 


5) 调用processNoCache()方法
读取ControllerConfig对象的nocache属性,如果nocache属性为true,在响应结果中将加入特定的头参数:Pragma,Cache-Control和Expires,
防止页面被存储在客户的浏览器的缓存中,processNoCache方法的代码如下:

protected void processNoCache(HttpServletRequest request,   
	                                  HttpServletResponse response) {   
	  
	        if (moduleConfig.getControllerConfig().getNocache()) {   
	            response.setHeader("Pragma", "No-cache");   
	            response.setHeader("Cache-Control", "no-cache,no-store,max-age=0");   
	            response.setDateHeader("Expires", 1);   
	        }   
	    }  



6)调用processPreprocess()方法
该方法不执行任何操作.直接返回true.子类可以覆盖这个方法.
执行客户化的预处理请求操作.

7)调用processMapping()方法

寻找和用户请求的URI匹配的ActionMapping,如果不存在这样的ActionMapping,则向用户返回恰当的错误信息.

8)调用processRoles()方法
先判断是否为Action配置了安全角色,如果配置了安全角色,就调用isUserInRole()方法判断当前用户是否具备必需的角色,如果不具备,就结束请求处理流程.,向用户返回恰当的错误消息.

9)调用processActionForm()方法
先判断是否为ActionMapping配置了ActionForm,如果配置了ActionForm,就先从ActionForm的存在范围内(request或session)寻找改ActionForm实例,如果不存在,就创建一个实例,接下来把它保存在合适的范围内,保存时使用的属性key为ActionMapping的name属性。

10)调用processPopulate()方法
如果为ActionMapping配置了ActionForm,就先调用ActionForm的reset()方法,再把请求中的表单数据组装到ActionForm中。

11)调用processValidate()方法
如果为ActionMapping配置了ActionForm,并且ActionMapping的validate属性为true,就调用ActionForm的validate()方法,如果validate方法返回的ActionErrors对象中包含ActionMessage对象,说明表单验证失败。就把ActionErrors对象放在request范围内,再把请求转发到ActionMapping的input属性指定的Web组件。如果ActionForm的validate方法执行表单验证成功,就继续执行下面的处理流程。

12)调用processForward()方法
判断是否在ActionMapping中配置了forward属性。如果配置了这个属性,就调用RequestDispatcher的forward方法,请求处理流程结束。否则进行下一步。

13)调用processInclude()方法
判断是否在ActionMapping中配置了include属性。如果配置了这个属性,就调用RequestDispatcher的include方法,请求处理流程结束。否则进行下一步。

14)调用processActionCreate()方法
先判断是否在Action缓存中存在这个Action实例,如果没有就新建一个Action实例,把它放在Action缓存中。可以看出Action也是只有一个实例在运行的。

15)调用processActionPerform
该方法调用Action实例的execute方法,该方法位于try/catch中,以及捕获异常。processActionPerform()方放代码如下。
	protected ActionForward   
	       processActionPerform(HttpServletRequest request,   
	                            HttpServletResponse response,   
	                            Action action,   
	                            ActionForm form,   
	                            ActionMapping mapping)   
	       throws IOException, ServletException {   
	       try {   
	           return (action.execute(mapping, form, request, response));   
	       } catch (Exception e) {   
	           return (processException(request, response,   
	                                    e, form, mapping));   
	       }   
	   



16)调用processActionForward方法
把你的Action的excute方法返回的ActionFoward对象作为参数传给它,processActionForward对象包的请求转发信息来执行请求转发或重定向。

在RequestProcessor类的process方法中,会访问ControllerConfig、ActionMappig和ActionForward实力的属性,ControllerConfig类和struts配置文件的<controlle>r元素对应,ActionMapping类和<action>元素对应,ActionForward和<forward>元素对应,process方法通过访问这三个类实例的属性来获得相关的配置信息。
写了这么多,RequestProcessor干得事够多的吧。
二、扩展RequestProcessor类

如果想修改RequestProcessor的一些默认功能,改易覆盖RequestProcessor基类中的相关方法.

Public class CustomRequestProcessor extends RequestProcessor{   
	  protected void processPreprocess (HttpServletRequest request,   
	                                 HttpServletResponse response) {    
	………………….   
	}   
	}  


在struts配置文件中,<controller>元素的processorClass属性用于配置你自己的RequestProcessor类

</controller    
	contentType=“text/html:charset=”GB2312”   
	locale=”true” nocache=”true” processorCalss=”com.test.CustomRequestProcessor”/>  


分享到:
评论

相关推荐

    struts1整合spring 通过代理RequestProcessor实现

    这只是一个最简单的demo,只有一个action, 通过代理RequestProcessor实现

    Spring中集成Struts的三种方式

    //第一种方式:使用 Spring 的 ActionSupport 类整合 Structs ...//第二种方式:使用 Spring 的 DelegatingRequestProcessor 覆盖 Struts 的 RequestProcessor //第三种方式:将 Struts Action 管理委托给 Spring 框架

    精通Struts基于MVC的Java Web设计与开发 孙卫琴 光盘

    第4章到第7章深入探讨了Struts框架的核心组件ActionServlet和RequestProcessor的实现原理,详细介绍了开发Struts应用的模型、视图和控制器的各种技术,细致的描述了Struts配置文件的每个元素的使用方法。第8章到第9...

    JSP Struts配置文件详解

    默认org.apache.struts.action.RequestProcessor @tempDir:指定文件上传时的临时工作目录.如果没有设置,将才用Servlet容器为web应用分配的临时工作目录. @nochache:true时,在相应结果中加入特定的头参数:Pragma ,...

    精通 Struts:基于 MVC 的 JavaWeb 设计与开发(PDF)

    第4章到第7章深入探讨了Struts框架的核心组件ActionServlet和RequestProcessor的实现原理,详细介绍了开发Struts应用的模型、视图和控制器的各种技术,细致的描述了Struts配置文件的每个元素的使用方法。第8章到第9...

    Struts1.3 备忘笔记

    05 Struts_05DynaActionForm : 相对应ActionForm来说,动态ActionForm只需要在配置文件中配置就可以了,但是它不能实现验证功能,演示人机交互实例。 06 Struts_06Filter : 中文编码的统一处理,通过Struct的请求...

    精通Struts_基于MVC的Java Web设计与开发

    第4章到第7章深入探讨了Struts框架的核心组件ActionServlet和RequestProcessor的实现原理,详细介绍了开发Struts应用的模型、视图和控制器的各种技术,细致的描述了Struts配置文件的每个元素的使用方法。第8章到第9...

    精通struts:基于mvc的java web设计与开发part3

    第4章到第7章深入探讨了Struts框架的核心组件ActionServlet和 RequestProcessor的实现原理,详细介绍了开发Struts应用的模型、视图和控制器的各种技术,细致的描述了Struts配置文件的每个元素的使用方法。...

    精通struts:基于mvc的java web设计与开发part2

    第4章到第7章深入探讨了Struts框架的核心组件ActionServlet和 RequestProcessor的实现原理,详细介绍了开发Struts应用的模型、视图和控制器的各种技术,细致的描述了Struts配置文件的每个元素的使用方法。...

    无意间发现的sturts过滤器

    Struts过滤器可以解决页面之间的乱码问题 一个类继承struts的RequestProcessor(分request 和response处理) 然后在struts-config.xml配置即可!

    精通struts:基于mvc的java web设计与开发part1

    第4章到第7章深入探讨了Struts框架的核心组件ActionServlet和 RequestProcessor的实现原理,详细介绍了开发Struts应用的模型、视图和控制器的各种技术,细致的描述了Struts配置文件的每个元素的使用方法。...

    struts1流程和原理

    struts1的几个核心组件是值得我们注意的: 1 、ActionServlet(核心控制器)。 2、RequestProcessor类(处理异常的核心组件)。 3、ActionForm(接收页面中传过的数据)。 4、Action(是控制器,主要是从ActionForm中...

    Struts开发指南之工作流程

     ActionMapping是ActionConfig的子类,实质上是对struts-config.xml的一个映射,从中可以取得所有的配置信息RequestProcessor根据提交过来的url,如*.do,从ActionMapping中得到相应的ActionForn和Action。...

    Struts控制器示例

    Struts的控制器主要包括: ActionServlet :核心控制器 RequestProcessor :子应用模块的处理器 Action :处理单项业务

    Hibernate+Spring+Struts扩展Struts

    简介: 我看到很多项目中,开发者实现了自己的MVC框架,并不是因为他们想做同Struts根本不同的东西,而是因为他们并没有意识到如何扩展Struts。... 2、RequestProcessor:如果你想在请求被处理的过程中某个时

    基于MVC的java Web设计与开发

    第4章到第7章深入探讨了Struts框架的核心组件ActionServlet和RequestProcessor的实现原理,详细介绍了开发Struts应用的模型、视图和控制器的各种技术,细致的描述了Struts配置文件的每个元素的使用方法。第8章到第9...

    StrutsAPI(java)

    在請求來臨後,RequestProcessor 根據請求URI呼叫對應的Action物件,將工作交給它,並在最後由Action物件得到一個ActionForward物件, ActionServlet使用ActionForward得知將流程forward至指定的資源。 當請求到達...

    Struts_config.xml详解

    roles="" //指定允许调用该Action的安全角色,多个角色之间用,隔开,在处理请求时,RequestProcessor会根据该配置项来决定用户是否有权限调用Action权限。 scope="request" //指定ActionForm Bean的存在范围,可...

Global site tag (gtag.js) - Google Analytics