首页 > 其他 > 详细

struts2-invocation

时间:2014-02-19 21:00:06      阅读:302      评论:0      收藏:0      [点我收藏+]

[转载]       

“将Web页面中的输入元素封装为一个(请求)数据对象”,这个对象就是ActionInvocation类型.
        对于Xwork 而言,前端的Webwork 组件为其提供的是一个Map 类型的数据结构。而Action面向的却是Model对象所提供的数据结构。在何时、何处对这两种不同的数据结构进行转换?
        写一个辅助类完成这样的工作,并在每次Action 调用之前由框架代码调用他完成转换工作。
Xwork 通过Interceptor 实现了这一步骤,从而我们可以根据需要,灵活的配置所需的Interceptor。从而为Action提供可扩展的预处理、后处理过程。

          ActionInvocation 是Xworks 中Action 调度的核心。而对Interceptor 的调度,也正是由ActionInvocation负责。
         ActionInvocation 是一个接口, 而DefaultActionInvocation 则是Webwork 对ActionInvocation的默认实现。

      Interceptor 的调度流程大致如下:

      1. ActionInvocation初始化时,根据配置,加载Action相关的所有Interceptor。

      参见ActionInvocation.init方法中相关代码:
     

bubuko.com,布布扣private void init() throws Exception {
bubuko.com,布布扣……
bubuko.com,布布扣List interceptorList 
= new
bubuko.com,布布扣ArrayList(proxy.getConfig().getInterceptors());
bubuko.com,布布扣interceptors 
= interceptorList.iterator();
bubuko.com,布布扣



   2. 通过ActionInvocation.invoke方法调用Action实现时,执行Interceptor:
       下面是DefaultActionInvocation中Action调度代码:

bubuko.com,布布扣public String invoke() throws Exception {
bubuko.com,布布扣    
if (executed)
bubuko.com,布布扣        
throw new IllegalStateException("Action has already executed");
bubuko.com,布布扣    
if (interceptors.hasNext()) {
bubuko.com,布布扣        Interceptor interceptor 
= (Interceptor) interceptors.next();
bubuko.com,布布扣        resultCode 
= interceptor.intercept(this);
bubuko.com,布布扣    }
 else
bubuko.com,布布扣        resultCode 
= invokeAction(getAction(), proxy.getConfig());
bubuko.com,布布扣    
if (!executed) {
bubuko.com,布布扣        
if (preResultListeners != null{
bubuko.com,布布扣        Iterator iterator 
= preResultListeners.iterator();
bubuko.com,布布扣        
while (iterator.hasNext()) {
bubuko.com,布布扣            PreResultListener listener
bubuko.com,布布扣            
= (PreResultListener) iterator.next();
bubuko.com,布布扣            listener.beforeResult(
this, resultCode);
bubuko.com,布布扣        }

bubuko.com,布布扣        }

bubuko.com,布布扣        
if (proxy.getExecuteResult())
bubuko.com,布布扣        executeResult();
bubuko.com,布布扣        executed 
= true;
bubuko.com,布布扣    }

bubuko.com,布布扣    
return resultCode;
bubuko.com,布布扣    }

bubuko.com,布布扣    



所有的拦截器都必须实现Interceptor 接口。

      public interface Interceptor {
         void destroy();
         void init();
         String intercept(ActionInvocation invocation) throws Exception;
     }
     在Interceptor 实现中,抽象实现AroundInterceptor得到了最广泛的应用(扩展),它增加了预处理(before)和后处理(after)方法的定义。

     AroundInterceptor.java:

bubuko.com,布布扣public abstract class AroundInterceptor implements Interceptor
bubuko.com,布布扣
{
bubuko.com,布布扣    
protected Log log = LogFactory.getLog(this.getClass());
bubuko.com,布布扣    
bubuko.com,布布扣    
public void destroy() {
bubuko.com,布布扣    }

bubuko.com,布布扣    
bubuko.com,布布扣    
public void init() {
bubuko.com,布布扣    }

bubuko.com,布布扣    
bubuko.com,布布扣    
public String intercept(ActionInvocation invocation) throws Exception {
bubuko.com,布布扣    String result 
= null;
bubuko.com,布布扣    before(invocation);
bubuko.com,布布扣    result 
= invocation.invoke();
bubuko.com,布布扣    after(invocation, result);
bubuko.com,布布扣    
return result;
bubuko.com,布布扣    }

bubuko.com,布布扣    
bubuko.com,布布扣    
protected abstract void after
bubuko.com,布布扣    (ActionInvocation actioninvocation, String string) 
throws Exception;
bubuko.com,布布扣    
bubuko.com,布布扣    
protected abstract void before(ActionInvocation actioninvocation)
bubuko.com,布布扣    
throws Exception;
bubuko.com,布布扣}


    AroundInterceptor.invoke 方法中,调用了参数invocation的invoke 方法。

 最后,结合最常用的ParametersInterceptor,看看Xwork 是如何通过Interceptor,将Webwork传入的Map类型数据结构,转换为Action所需的Java 模型对象。

   ParametersInterceptor.java:

bubuko.com,布布扣public class ParametersInterceptor extends AroundInterceptor {
bubuko.com,布布扣
protected void after(ActionInvocation dispatcher, String result)
bubuko.com,布布扣
throws Exception {
bubuko.com,布布扣}

bubuko.com,布布扣
protected void before(ActionInvocation invocation) throws Exception
bubuko.com,布布扣
{
bubuko.com,布布扣
if (!(invocation.getAction() instanceof NoParameters)) {
bubuko.com,布布扣
final Map parameters =
bubuko.com,布布扣ActionContext.getContext().getParameters(); ⑴
bubuko.com,布布扣
if (log.isDebugEnabled()) {
bubuko.com,布布扣log.debug(
"Setting params " + parameters);
bubuko.com,布布扣}

bubuko.com,布布扣ActionContext invocationContext 
=
bubuko.com,布布扣invocation.getInvocationContext();
bubuko.com,布布扣
try {
bubuko.com,布布扣invocationContext.put(
bubuko.com,布布扣InstantiatingNullHandler.CREATE_NULL_OBJECTS,
bubuko.com,布布扣Boolean.TRUE);
bubuko.com,布布扣invocationContext.put(
bubuko.com,布布扣XWorkMethodAccessor.DENY_METHOD_EXECUTION,
bubuko.com,布布扣Boolean.TRUE);
bubuko.com,布布扣invocationContext.put(
bubuko.com,布布扣XWorkConverter.REPORT_CONVERSION_ERRORS,
bubuko.com,布布扣Boolean.TRUE);
bubuko.com,布布扣
if (parameters != null{
bubuko.com,布布扣
final OgnlValueStack stack =
bubuko.com,布布扣ActionContext.getContext().getValueStack(); ⑵
bubuko.com,布布扣
for (Iterator iterator =parameters.entrySet().iterator();
bubuko.com,布布扣iterator.hasNext();
bubuko.com,布布扣
{
bubuko.com,布布扣Map.Entry entry 
= (Map.Entry) iterator.next();
bubuko.com,布布扣stack.setValue( ⑷
bubuko.com,布布扣entry.getKey().toString(),
bubuko.com,布布扣entry.getValue());
bubuko.com,布布扣}

bubuko.com,布布扣}

bubuko.com,布布扣}
 finally {
bubuko.com,布布扣invocationContext.put(
bubuko.com,布布扣InstantiatingNullHandler.CREATE_NULL_OBJECTS,
bubuko.com,布布扣Boolean.FALSE);
bubuko.com,布布扣invocationContext.put(
bubuko.com,布布扣XWorkMethodAccessor.DENY_METHOD_EXECUTION,
bubuko.com,布布扣Boolean.FALSE);
bubuko.com,布布扣invocationContext.put(
bubuko.com,布布扣XWorkConverter.REPORT_CONVERSION_ERRORS,
bubuko.com,布布扣Boolean.FALSE);
bubuko.com,布布扣}

bubuko.com,布布扣}

bubuko.com,布布扣}

bubuko.com,布布扣}



ParametersInterceptor 扩展了抽象类AroundInterceptor。并在其预处理方法(before)中实现了数据的转换。
数据转换的过程并不复杂:
⑴ 首先由ActionContext获得Map型的参数集parameters。
⑵ 由ActionContext获得值栈(OgnlValueStack)。
⑶ 遍历parameters中的各项数据。
⑷ 通过OgnlValueStack,根据数据的键值,为Model 对象填充属性数据。
OgnlValueStack 是http://www.ognl.org4提供的一套可读写对象属性的类库

上面的代码中并没有发现将Model对象入栈的部分,是由于ActionInvocation在初始化的时候已经预先完成了压栈工作,如DefaultActionInvocation.init方法中代码所示:

private void init() throws Exception {
Map contextMap = createContextMap();
createAction();
if (pushAction) {
stack.push(action);      //压栈
}
……


旁白:Servlet 2.3规范中引入的Filter 算是拦截器的一个典型实现,它在Servlet执行之前被触发,对输入参数进行处理之后,再将工作流程传递给对应的Servlet。

struts2-invocation

原文:http://www.cnblogs.com/wangjianbg/p/3555584.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!