ChainingInterceptor

该拦截器处于defaultStack第六的位置,其主要功能是复制值栈(ValueStack)中的所有对象的所有属性到当前正在执行的Action中,如果说ValueStack中没有任何对象的话,该拦截器不会干任何事情,看到这个拦截器的名称,大家应该会想到有一种chain类型的Result,该拦截器主要就是针对chain类型Result起作用的,因为我们有可能在chain链后面的Action用到前面Action的属性,所以struts2提供了该拦截器来实现这个功能。当然我们也可以让chain链中的某个Action属性不复制到正在执行的Action中,只要chain链中的Action实现com.opensymphony.xwork2.Unchainable接口,这个Action中的属性就不会复制到当前正在执行的Action中了。下面我们看一个该拦截器的源码:

@Override
public String intercept(ActionInvocation invocation) throws Exception {
ValueStack stack = invocation.getStack();//获取值栈
CompoundRoot root = stack.getRoot();//获取值栈中的root对象,其实就是一个List
//如果值栈中有对象
if (root.size() > 1) {
//把值栈中的对象引用复制到一个List中,目的是不破坏值栈中的对象
List<CompoundRoot> list = new ArrayList<CompoundRoot>(root);
list.remove(0);//移除第一个元素,即当前正在执行的Action对象
//然后将该List中对象反序,这样复制到当前Action中的属性是chain链中最后面Action的属性
Collections.reverse(list);
//获取ActionContext对象内部的contextMap对象,其实就是OgnlContext对象
Map<String, Object> ctxMap = invocation.getInvocationContext().getContextMap();
Iterator<CompoundRoot> iterator = list.iterator();
int index = 1; // starts with 1, 0 has been removed
while (iterator.hasNext()) {//迭代
index = index + 1;
Object o = iterator.next();
if (o != null) {
if (!(o instanceof Unchainable)) {//如果当前被迭代的对象没有实现Unchainable接口,则要进行属性复制
reflectionProvider.copy(o, invocation.getAction(), ctxMap, excludes, includes);//复制属性
}
}
else {
LOG.warn("compound root element at index "+index+" is null");
}
}
} return invocation.invoke();
}

注释已经写得清楚了,下面以一段Action配置来进一步讲解,配置如下:

<action name="test" class="com.xtayfjpk.action.TestAction">
<result name="chain" type="chain">chain</result>
</action>
<action name="chain" class="com.xtayfjpk.action.ChainAction">
<result name="success">/WEB-INF/page/message.jsp</result>
</action>

这里test action的Result类型为chain,指向了chain action,当执行TestAction时,值栈中就两个对象,第一个是TestAction自己,第二个是一个com.opensymphony.xwork2.DefaultTextProvider对象,由于当前Action被移除不会参数迭代,而DefaultTextProvider实现了Unchainable接口,所以不会执行reflectionProvider.copy方法,也就没有属性的复制。
   当执行ChainAction到该拦截器时,值栈中有三个对象,第一个是ChainAction自己,第二个是TestAction对象,第三个是DefaultTextProvider,
根据上面的逻辑,就会把TestAction的所有属性复制给ChainAction对象,如果说ChainAction返回的又是chain类型的Result,则又会把ChainAction和TestAction的所有属性复制给下一个Action,依此类推......

现在我们看一下这句属性复制代码:reflectionProvider.copy(o, invocation.getAction(), ctxMap, excludes, includes);copy方法的第一个参数就是值栈中要被复制属性的对象,第二个参数是当前正在执行的Action,第三个参数是OgnlContext对象,第四、五个参数excludes与includes是一个属性名称集合,excludes表示不需要被复制的属性集合,includes表示要被复制的属性集合,下面是copy方法源码:

public void copy(Object from, Object to, Map<String, Object> context,
Collection<String> exclusions, Collection<String> inclusions) {
ognlUtil.copy(from, to, context, exclusions, inclusions);
}

它调用了ognlUtil.copy方法,我们进入该方法:

public void copy(Object from, Object to, Map<String, Object> context, Collection<String> exclusions, Collection<String> inclusions) {
//省略... PropertyDescriptor[] fromPds;
PropertyDescriptor[] toPds; try {
fromPds = getPropertyDescriptors(from);//获取源对象(要被复制属性的对象)的属性描述符
toPds = getPropertyDescriptors(to);//获目标对象(当前正在执行的Action对象)的属性描述符
} catch (IntrospectionException e) {
LOG.error("An error occured", e); return;
} Map<String, PropertyDescriptor> toPdHash = new HashMap<String, PropertyDescriptor>(); for (PropertyDescriptor toPd : toPds) {
toPdHash.put(toPd.getName(), toPd);//将目标对象属性描述符放到一个HashMap中
}
//迭代获取源对象属性描述符
for (PropertyDescriptor fromPd : fromPds) {
if (fromPd.getReadMethod() != null) {//当前属性的读方法不为空
boolean copy = true;
if (exclusions != null && exclusions.contains(fromPd.getName())) {
//如果当前属性包含在exclusions集合中则不复制
copy = false;
} else if (inclusions != null && !inclusions.contains(fromPd.getName())) {
//如果配置了inclusions,而当前属性又不包含在inclusions集合中则不复制
//如果没有配置inclusions,则只要不包含在exclusions就会复制
copy = false;
} if (copy == true) {//如果当前属性需要复制
PropertyDescriptor toPd = toPdHash.get(fromPd.getName());//获取目标对象相应的属性描述符
if ((toPd != null) && (toPd.getWriteMethod() != null)) {//如果属性描述符存在且其写方法不为空则进行复制
try {
Object expr = compile(fromPd.getName());
Object value = Ognl.getValue(expr, contextFrom, from);
Ognl.setValue(expr, contextTo, to, value);//真正复制属性的代码
} catch (OgnlException e) {
// ignore, this is OK
}
} } } }
}

至此,该拦截器要复制哪些属性,要复制哪些对象的属性,以及复制属性的具体规则已经讲解完毕了,最后一句:Ognl.setValue(expr, contextTo, to, value);
真正复制属性的代码已经涉及到OGNL内部的实现机制,这是很复杂的,但对于理解该拦截器来说不是很重要,这里就不说了,有兴趣的可自行研究。

现在回到intercept方法,当属性复制完成后调用invocation.invoke();继续执行下一个拦截器......

struts2 18拦截器详解(七)的更多相关文章

  1. struts2 18拦截器详解(十)

    ModelDrivenInterceptor 该拦截器处于defaultStack中的第九的位置,在ScopedModelDrivenInterceptor拦截器之后,要使该拦截器有效的话,Actio ...

  2. struts2 18拦截器详解(九)

    ScopedModelDrivenInterceptor 该拦截器处于defaultStack第八的位置,其主要功能是从指定的作用域内检索相应的model设置到Action中,该类中有三个相关的属性: ...

  3. struts2 18拦截器详解(五)

    I18nInterceptor 该拦截器处理defaultStack第四的位置,是用来方便国际化的,如果说我们的一个Web项目要支持国际化的话,通常的做法是给定一个下拉框列出所支持的语言,当用户选择了 ...

  4. 通俗易懂之SpringMVC&Struts2前端拦截器详解

    直接进入主题吧!一,配置Struts2的拦截器分两步走1配置对应的拦截器类:2在配置文件Struts.xml中进行配置拦截器同时在Strust2中配置拦截器类有三种方法1实现Interceptor接口 ...

  5. struts2内置拦截器和自定义拦截器详解(附源码)

    一.Struts2内置拦截器 Struts2中内置类许多的拦截器,它们提供了许多Struts2的核心功能和可选的高级特 性.这些内置的拦截器在struts-default.xml中配置.只有配置了拦截 ...

  6. Struts2 之 modelDriven & prepare 拦截器详解

    struts2 ModelDriven & Prepareable 拦截器 前面对于 Struts2 的开发环境的搭建.配置文件以及 Struts2 的值栈都已经进行过叙述了!这次博文我们讲解 ...

  7. Struts2拦截器详解

    一.Struts2拦截器原理: Struts2拦截器的实现原理相对简单,当请求struts2的action时,Struts 2会查找配置文件,并根据其配置实例化相对的    拦截器对象,然后串成一个列 ...

  8. Struts2中的拦截器详解

    exception:异常拦截器,拦截异常aliasservletConfig18nprepare:预备拦截器,这个拦截器就是为了ModelDriven准备对象的,若Action类实现了preparab ...

  9. [转]SpringMVC拦截器详解[附带源码分析]

      目录 前言 重要接口及类介绍 源码分析 拦截器的配置 编写自定义的拦截器 总结 前言 SpringMVC是目前主流的Web MVC框架之一. 如果有同学对它不熟悉,那么请参考它的入门blog:ht ...

随机推荐

  1. Sql server中内连接语句

    数据库中学生表和课程表如下: 内连接sql语句: select a.studentName,a.studentAge,b.courseName from student a inner join co ...

  2. 利用Aspose.Word控件实现Word文档的操作

    Aspose系列的控件,功能都挺好,之前一直在我的Winform开发框架中用Aspose.Cell来做报表输出,可以实现多样化的报表设计及输出,由于一般输出的内容比较正规化或者多数是表格居多,所以一般 ...

  3. Tuning SQL via case when statement

    原SQL如下:SQL的主要问题是红色部分居然通过标量查询,反复的查找与SQL相同的基表,很显然这个可以用case when来简化. select a.TRAN_ID,a.AMOUNT,a.BALANC ...

  4. 使用jQuery为表单添加回车事件

    $(document).keypress(function(e){ if(e.which==13){ checkUserForm(); } });

  5. connect调用超时的实现方式

    第二种更通用的.使connect调用超时的方法是使套接字成为无阻塞的,然后用select等待它完成.这种方法避免了使用alarm时遇到的很多问题,但我们必须承认,即使是在UNIX实现中,这种方法还是存 ...

  6. EF Code First教程-01 创建一个简单的Code First程序

    1 从nuget中搜索并添加EF 2 在app.config或web.config中添加数据库连接 <connectionStrings> <add name="conns ...

  7. 1、JavaScript入门篇

    一.你知道,为什么JavaScript非常值得我们学习吗? 1. 所有主流浏览器都支持JavaScript. 2. 目前,全世界大部分网页都使用JavaScript. 3. 它可以让网页呈现各种动态效 ...

  8. 第九篇 SQL Server安全透明数据加密

    本篇文章是SQL Server安全系列的第九篇,详细内容请参考原文. Relational databases are used in an amazing variety of applicatio ...

  9. ARC机制集合内存管理

    // //  main.m //  13-ARC机制集合内存管理 // //  Created by apple on 14-3-21. //  Copyright (c) 2014年 apple. ...

  10. sh.status()

    mongos> sh.status({"verbose":1})sh.status({"verbose":1}) --- Sharding Status ...