当前位置:编程学习 > JS >>

struts2 18拦截器详解(六)

PrepareInterceptor
 
   该拦截器处于defaultStack第五的位置,该拦截器的功能主要是在Action的execute(假设配置文件中没有指定)方法之前执行一些业务逻辑,如果你刚好有这样的需求,该拦截器是你很好的选择,要使该拦截有效,Action要实现Preparable接口,与前面几个拦截器不同的是该拦截器继承自MethodFilterInterceptor而不是直接继承自AbstractInterceptor,MethodFilterInterceptor可以对拦截器的方法进行过滤,MethodFilterInterceptor拦截器将在AnnotationValidationInterceptor拦截器时一起讲解,暂时就可以理解为与直接继承自AbstractInterceptor无异,该拦截器中的doIntercept方法理解为以前的intercept方法,是处理功能逻辑的地方。下面是doIntercept方法的源码:
 
[java]  
@Override  
public String doIntercept(ActionInvocation invocation) throws Exception {  
    Object action = invocation.getAction();  
  
    if (action instanceof Preparable) {//判断当前Action是否实现了Preparable接口  
        try {//如果实现了调用前缀方法  
            PrefixMethodInvocationUtil.invokePrefixMethod(invocation,  
                    new String[]{PREPARE_PREFIX, ALT_PREPARE_PREFIX});  
        }  
        catch (InvocationTargetException e) {  
            LOG.warn("an exception occured while trying to execute prefixed method", e);  
        }  
        catch (IllegalAccessException e) {  
            LOG.warn("an exception occured while trying to execute prefixed method", e);  
        } catch (Exception e) {  
            LOG.warn("an exception occured while trying to execute prefixed method", e);  
        }  
        //是否总是要执行prepare方法,alwaysInvokePrepare的默认值为true  
        if (alwaysInvokePrepare) {  
            ((Preparable) action).prepare();  
        }  
    }  
    return invocation.invoke();//调用下一个拦截器  
}  
 
   该方法中逻辑很简单,主要逻辑在PrefixMethodInvocationUtil.invokePrefixMethod方法中,执行时把ActionInvocation对象与两个前缀字符串放在一个数组中传给了该方法,两个前缀分别是"prepare","prepareDo",这两个值是在该拦截器中写死了的。现在去看一下这个方法的源码:
 
[java] 
public static void invokePrefixMethod(ActionInvocation actionInvocation, String[] prefixes) throws InvocationTargetException, IllegalAccessException {  
    Object action = actionInvocation.getAction();//获取当前执行的Action  
    //获取Action要执行的方法名称  
    String methodName = actionInvocation.getProxy().getMethod();  
    //如果方法名为null则说明在struts2的配置文件中没有配置Action要执行的方法名称  
    if (methodName == null) {  
        // 如果为null则表示执行execute方法,DEFAULT_INVOCATION_METHODNAME值就是"execute"  
        methodName = DEFAULT_INVOCATION_METHODNAME;  
    }  
    //获取前缀方法  
    Method method = getPrefixedMethod(prefixes, methodName, action);  
    if (method != null) {//如果不为null则调用前缀方法  
        method.invoke(action, new Object[0]);  
    }  
}  
 
   下面我们去看一下getPrefixedMethod(prefixes, methodName, action);,看它是如何获取前缀方法的:
[java] 
public static Method getPrefixedMethod(String[] prefixes, String methodName, Object action) {  
    assert(prefixes != null);  
    //把方法名称的第一个字母大写并返回,这个可以自己参看一下源码  
    String capitalizedMethodName = capitalizeMethodName(methodName);  
    for (String prefixe : prefixes) {//迭代前缀数组  
        //这里就是将要执行的方法名首字母大后前面加上相应的前缀(prepare或prepareDo)  
        String prefixedMethodName = prefixe + capitalizedMethodName;  
        try {//返回该方法,prepare或prepareDo中可有一个生效  
            return action.getClass().getMethod(prefixedMethodName, EMPTY_CLASS_ARRAY);  
        }  
        catch (NoSuchMethodException e) {  
            // hmm -- OK, try next prefix  
            if (LOG.isDebugEnabled()) {  
                LOG.debug("cannot find method [" + prefixedMethodName + "] in action [" + action + "]");  
            }  
        }  
    }  
    return null;  
}  
 
   到这里前缀方法的获取与调用过程已经讲解完成,然后就是判断是否总是要执行prepare方法,如果alwaysInvokePrepare的值为true则会执行Action的prepare方法,prepare方法就是在Preparable接口中声明的方法。
补充:Web开发 , Jsp ,
CopyRight © 2012 站长网 编程知识问答 www.zzzyk.com All Rights Reserved
部份技术文章来自网络,