ScopedModelDrivenInterceptor

该拦截器处于defaultStack第八的位置,其主要功能是从指定的作用域内检索相应的model设置到Action中,该类中有三个相关的属性:
scope:从哪里获取model,有两个值,一是request,二是session,默认值为request
name:在scope中检索model的key
className:model类的名称,也就是该model是什么类型

虽然这三个参数都可以进行设置,但是在defaultStack中struts2并没有为其赋值,所以初始值这三个属性都为null。

下面我们去看一个ScopedModelDriven接口,该接口继承自ModelDriven接口,实现了ScopedModelDriven接口也就实现了ModelDriven,所以当前Action也会有ModelDriven的特性,ModelDriven在下一个拦截器中介绍。ScopedModelDriven接口中相当于有四个方法:
1.getModel() -->用于Action获取model
2.setModel() -->提供给ScopedModelDrivenInterceptor为Action设置model
3.getScopeKey() -->用于Action获取model存储在作用域中的key,这里对应了ScopedModelDrivenInterceptor中的name属性
4.setScopeKey() -->提供给ScopedModelDrivenInterceptor为Action设置model存储的key

下面是ScopedModelDrivenInterceptor的intercept方法源码:

@Override
public String intercept(ActionInvocation invocation) throws Exception {
Object action = invocation.getAction();//获取当前正在执行的Action
//如果Action实现了ScopedModelDriven接口
if (action instanceof ScopedModelDriven) {
ScopedModelDriven modelDriven = (ScopedModelDriven) action;//强转
if (modelDriven.getModel() == null) {//如果返回为null,如果不为null的话是不会去修改该Action中model属性的
ActionContext ctx = ActionContext.getContext();//获取ActionContext
ActionConfig config = invocation.getProxy().getConfig();//获取Action配置
//model类名,初始值为null
String cName = className;
if (cName == null) {
try {
Method method = action.getClass().getMethod(GET_MODEL, EMPTY_CLASS_ARRAY);//通过反射获取getModel方法
Class cls = method.getReturnType();//获取getModel方法的返回值类型
cName = cls.getName();//获取model类名
} catch (NoSuchMethodException e) {//如果没有getModel方法则抛异常
throw new XWorkException("The " + GET_MODEL + "() is not defined in action " + action.getClass() + "", config);
}
}
String modelName = name;//model存储的key,初始值为null
if (modelName == null) {
modelName = cName;//把model的类名赋给model存储的key
}
Object model = resolveModel(objectFactory, ctx, cName, scope, modelName);//解析model
modelDriven.setModel(model);//把返回的model对象设置到Action中
modelDriven.setScopeKey(modelName);//返回model存储的key设置到Action中
}
}
return invocation.invoke();//调用下一个拦截器
}

该方法的逻辑还是挺简单的,如果Action的getModel方法返回不为null的话,strust2就不会去修改Action的model属性,这也更符合我们的应用
要求,不然这个拦截器就太"霸道"了,然后去解析获取model对象,最后把返回的model对象与model对象存储的key设置到Action中,现在去看一下
resolveModel方法:

protected Object resolveModel(ObjectFactory factory, ActionContext actionContext, String modelClassName, String modelScope, String modelName) throws Exception {
Object model = null;
Map<String, Object> scopeMap = actionContext.getContextMap();//获取ActionContext内容的context Map对象
if ("session".equals(modelScope)) {//如果配置了scope为session,但这里并没有配置
scopeMap = actionContext.getSession();//所以这里不会执行
} model = scopeMap.get(modelName);//去context Map(即相当于request作用域)中查找
if (model == null) {//没有找到
model = factory.buildBean(modelClassName, null);//调用对象工厂的buildBean方法实例化一个对象
scopeMap.put(modelName, model);//放置到context Map中
}
return model;//返回model对象
}

该方法先去作用域查找,如果查找不到再由ObjectFactory对象创建一个,ObjectFactory是struts2中专门用于创建对象的,其内部就是通过反射调用
class.newInstance()创建,sturst2中的Action、Result、Interceptor都是由ObjectFactory创建。model对象创建好后放入到了request作用域中,该方法执行完成后又把返回的model设置进了Action,如果Action对model进行了接收,那么Action中的model是有值的。

到这该拦截器就讲解完毕了,最后调用invocation.invoke();调用下一个拦截器

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

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

    ChainingInterceptor 该拦截器处于defaultStack第六的位置,其主要功能是复制值栈(ValueStack)中的所有对象的所有属性到当前正在执行的Action中,如果说Valu ...

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

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

  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. Windows和linux下clock函数

    windows:  Calculates the wall-clock time used by the calling process. return:The elapsed wall-clock ...

  2. 使用CefSharp在.Net程序中嵌入Chrome浏览器(九)——性能问题

    在使用CEF的过程中,我发现了一个现象:WPF版的CEF比Chrome性能要差:一些有动画的地方会掉帧(例如,CSS动画,全屏图片拖动等),视频播放的效果也没有Chrome流畅. 查了一下相关资料,发 ...

  3. Make the DbContext Ambient with UnitOfWorkScope(now named DbContextScope by mehdime)

    The Entity Framework DbContext (or LINQ-to-SQL DataContext) is a Unit Of Work implementation. That m ...

  4. 利用Fiddler和Wireshark解密SSL加密流量

    原文地址:http://kelvinh.github.io/blog/2014/01/12/decrypt-ssl-using-fiddler-and-wireshark/ Fiddler是一个著名的 ...

  5. android iOS 编码问题害死人!

    android 与后端服务器进行通信时,默认使用的编码格式是asi. 而iOS与后端通信时,获取的数据到iOS端默认被utf-8进行编码.所以,我们常常出现android能够从服务器端获取到数据,但是 ...

  6. NSZombie 详解 -僵尸对象

    1.什么是僵尸对象? 简而言之,就是过度释放的对象. 2.僵尸对象有什么特点? 如果一个对象a被变成了僵尸对象,那么,在进行下面的判断时,a是会被系统当成一个对象来进行判断的.但是,如果你使用a进行其 ...

  7. nginx网站攻击防护

    1.上上个月架构全部迁移上云以后,总的来说比较稳定,业务量也上来,可爱的坏人也来了,7X24小时不停恶意攻击我的网站,第一次收到报警是网站流入流量1分钟以内连续3次超过1000000bps,换算下1M ...

  8. sql --- where concat

    // where cancat 函数 SELECT * from  users WHERE email != CONCAT(username, '', '@huan1234qiu.com');

  9. Android Lint简介

    Android Lint是SDK Tools 16 (ADT 16)之后才引入的工具,通过它对Android工程源代码进行扫描和检查,可发现潜在的问题,以便程序员及早修正这个问题.Android Li ...

  10. 得到view坐标的各种方法

    这篇文章讲的方法全是再控件可以获取焦点的情况下执行的,如果在oncreat()里面执行,那么得到的都是0 1.getLocationInWindow 这个方法得到的是view相对于当前Activity ...