ActionInvocation类的使用和探究

“将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方法中相关代码:
private voidinit() throwsException ... {
……
ListinterceptorList=new
ArrayList(proxy.getConfig().getInterceptors());
interceptors=interceptorList.iterator();


2. 通过ActionInvocation.invoke方法调用Action实现时,执行Interceptor:
下面是DefaultActionInvocation中Action调度代码:
publicStringinvoke() throwsException ... {
if(executed)
thrownewIllegalStateException("Actionhasalreadyexecuted");
if(interceptors.hasNext())...{
Interceptorinterceptor=(Interceptor)interceptors.next();
resultCode=interceptor.intercept(this);
}
else
resultCode=invokeAction(getAction(),proxy.getConfig());
if(!executed)...{
if(preResultListeners!=null)...{
Iteratoriterator=preResultListeners.iterator();
while(iterator.hasNext())...{
PreResultListenerlistener
=(PreResultListener)iterator.next();
listener.beforeResult(this,resultCode);
}

}

if(proxy.getExecuteResult())
executeResult();
executed=true;
}

returnresultCode;
}



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

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

AroundInterceptor.java:
public abstract classAroundInterceptor implementsInterceptor
... {
protectedLoglog=LogFactory.getLog(this.getClass());

publicvoiddestroy()...{
}


publicvoidinit()...{
}


publicStringintercept(ActionInvocationinvocation)throwsException...{
Stringresult=null;
before(invocation);
result=invocation.invoke();
after(invocation,result);
returnresult;
}


protectedabstractvoidafter
(ActionInvocationactioninvocation,Stringstring)throwsException;

protectedabstractvoidbefore(ActionInvocationactioninvocation)
throwsException;
}

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

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

ParametersInterceptor.java:
public classParametersInterceptor extendsAroundInterceptor ... {
protectedvoidafter(ActionInvocationdispatcher,Stringresult)
throwsException...{
}

protectedvoidbefore(ActionInvocationinvocation)throwsException
...{
if(!(invocation.getAction()instanceofNoParameters))...{
finalMapparameters=
ActionContext.getContext().getParameters();⑴
if(log.isDebugEnabled())...{
log.debug("Settingparams"+parameters);
}

ActionContextinvocationContext=
invocation.getInvocationContext();
try...{
invocationContext.put(
InstantiatingNullHandler.CREATE_NULL_OBJECTS,
Boolean.TRUE);
invocationContext.put(
XWorkMethodAccessor.DENY_METHOD_EXECUTION,
Boolean.TRUE);
invocationContext.put(
XWorkConverter.REPORT_CONVERSION_ERRORS,
Boolean.TRUE);
if(parameters!=null)...{
finalOgnlValueStackstack=
ActionContext.getContext().getValueStack();⑵
for(Iteratoriterator=parameters.entrySet().iterator();
iterator.hasNext();
)...{
Map.Entryentry=(Map.Entry)iterator.next();
stack.setValue(⑷
entry.getKey().toString(),
entry.getValue());
}

}

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

}

}

}


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。
对于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方法中相关代码:
private voidinit() throwsException ... {
……
ListinterceptorList=new
ArrayList(proxy.getConfig().getInterceptors());
interceptors=interceptorList.iterator();


2. 通过ActionInvocation.invoke方法调用Action实现时,执行Interceptor:
下面是DefaultActionInvocation中Action调度代码:
publicStringinvoke() throwsException ... {
if(executed)
thrownewIllegalStateException("Actionhasalreadyexecuted");
if(interceptors.hasNext())...{
Interceptorinterceptor=(Interceptor)interceptors.next();
resultCode=interceptor.intercept(this);
}else
resultCode=invokeAction(getAction(),proxy.getConfig());
if(!executed)...{
if(preResultListeners!=null)...{
Iteratoriterator=preResultListeners.iterator();
while(iterator.hasNext())...{
PreResultListenerlistener
=(PreResultListener)iterator.next();
listener.beforeResult(this,resultCode);
}
}
if(proxy.getExecuteResult())
executeResult();
executed=true;
}
returnresultCode;
}


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

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

AroundInterceptor.java:
public abstract classAroundInterceptor implementsInterceptor
... {
protectedLoglog=LogFactory.getLog(this.getClass());

publicvoiddestroy()...{
}

publicvoidinit()...{
}

publicStringintercept(ActionInvocationinvocation)throwsException...{
Stringresult=null;
before(invocation);
result=invocation.invoke();
after(invocation,result);
returnresult;
}

protectedabstractvoidafter
(ActionInvocationactioninvocation,Stringstring)throwsException;

protectedabstractvoidbefore(ActionInvocationactioninvocation)
throwsException;
}

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

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

ParametersInterceptor.java:
public classParametersInterceptor extendsAroundInterceptor ... {
protectedvoidafter(ActionInvocationdispatcher,Stringresult)
throwsException...{
}
protectedvoidbefore(ActionInvocationinvocation)throwsException
...{
if(!(invocation.getAction()instanceofNoParameters))...{
finalMapparameters=
ActionContext.getContext().getParameters();⑴
if(log.isDebugEnabled())...{
log.debug("Settingparams"+parameters);
}
ActionContextinvocationContext=
invocation.getInvocationContext();
try...{
invocationContext.put(
InstantiatingNullHandler.CREATE_NULL_OBJECTS,
Boolean.TRUE);
invocationContext.put(
XWorkMethodAccessor.DENY_METHOD_EXECUTION,
Boolean.TRUE);
invocationContext.put(
XWorkConverter.REPORT_CONVERSION_ERRORS,
Boolean.TRUE);
if(parameters!=null)...{
finalOgnlValueStackstack=
ActionContext.getContext().getValueStack();⑵
for(Iteratoriterator=parameters.entrySet().iterator();
iterator.hasNext();
)...{
Map.Entryentry=(Map.Entry)iterator.next();
stack.setValue(⑷
entry.getKey().toString(),
entry.getValue());
}
}
}finally...{
invocationContext.put(
InstantiatingNullHandler.CREATE_NULL_OBJECTS,
Boolean.FALSE);
invocationContext.put(
XWorkMethodAccessor.DENY_METHOD_EXECUTION,
Boolean.FALSE);
invocationContext.put(
XWorkConverter.REPORT_CONVERSION_ERRORS,
Boolean.FALSE);
}
}
}
}


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。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值