Struts2拦截器之DefaultWorkflowInterceptor
一、DefaultWorkflowInterceptor是什么
首先说这东西是干嘛来的,在action中可以对传进来的数据进行验证,方法是实现Validateable接口的validate():void方法,如果发现错误可以addFieldError()或者addActionError(),当添加了错误之后struts2就不会再执行原本的映射方法,而是会直接返回一个结果字符串(如果正常的话返回的应该是要执行的映射到方法的返回值),我们这里要探讨的就是struts2是怎么知道我们添加了错误,又是怎么返回一个什么样的结果字符串呢?
二、DefaultWorkflowInterceptor源码分析
DefaultWorkflowInterceptor处于默认拦截器栈的最后(即第18个),主要是用来判断一下当前的action方法在执行过程中是否出现了错误,如果出现错误的话就返回一个表示错误的字符串同时打断拦截器执行栈。
下图是struts-default.xml文件中的默认拦截器栈这个类的配置信息:
需要注意的是对execludeMethods中指定的方法不执行DefaultWorkflowInterceptor拦截器。
DefaultWorkflowInterceptor的代码剖析:
网页看对齐、高亮效果不太好,建议粘贴到eclipse慢慢看 :-)
1 /**
2 * DefaultWorkflowInterceptor:
3 * 处于默认的拦截器栈最后的位置,用来判断Action的执行过程中有无错误
*
* **/
public class DefaultWorkflowInterceptor extends MethodFilterInterceptor { private static final long serialVersionUID = 7563014655616490865L; //日志
private static final Logger LOG = LoggerFactory.getLogger(DefaultWorkflowInterceptor.class); //酱油位,大概是为了避免每次都创建的话会很浪费吧(注意是static的哦)
private static final Class[] EMPTY_CLASS_ARRAY = new Class[0]; //表示要求重新输入的字符,当检查出问题的时候就返回这个字符串让用户重新输入
private String inputResultName = Action.INPUT; /**
* Set the <code>inputResultName</code> (result name to be returned when
* a action / field error is found registered). Default to {@link Action#INPUT}
*
* @param inputResultName what result name to use when there was validation error(s).
*/ //在配置的时候可以传入一个inputResultName的param,会通过这个方法设置进来的
public void setInputResultName(String inputResultName) {
this.inputResultName = inputResultName;
} /**
* Intercept {@link ActionInvocation} and returns a <code>inputResultName</code>
* when action / field errors is found registered.
*
* @return String result name
*/
@Override
protected String doIntercept(ActionInvocation invocation) throws Exception { //先获得当前的action对象
Object action = invocation.getAction(); //判断这个action是否实现了ValidationAware接口,意思就是看看当前的action需不需要验证
if (action instanceof ValidationAware) { //首先进行类型转换,类型转换后的action就会被当做ValidationAware处理了,下面就称这个action对象为ValidationAware实例
ValidationAware validationAwareAction = (ValidationAware) action; //如果这个ValidationAware实例中已经包含了错误信息,就进行一些操作然后把拦截器执行栈咔嚓掉,然后直接返回特定字串(默认是Action.INPUT)
if (validationAwareAction.hasErrors()) { //是否记录日志
if (LOG.isDebugEnabled()) {
LOG.debug("Errors on action " + validationAwareAction + ", returning result name 'input'");
} //默认的返回字串为Action.INPUT,是写死的,不过没关系我们后面可以通过实现ValidationWorkflowAware接口来自定义
String resultName = inputResultName; /*Action可以实现ValidationWorkflowAware然后实现getInputResultName():String方法来自定义当发生错误的时候的返回字串
比如实现不同的方法验证出处返回不同的字串,只需要在检验出错的时候设定一个成员变量的值然后在getInputResultName()返回其值就可以了,比如
在create方法中出错将事先声明的成员变量inputResultName设定为“create”,而在delete中出错将inputResultName设定为“delete”,这样就可以区分它们并可以跳转到不同的页面啦
但这只是一种折中的办法,一个良好的规范是ValidationWorkflowAware提供了方法级别的自定义返回字串,而@InputConfig提供了方法级别的自定义返回字串
*/
if (action instanceof ValidationWorkflowAware) {
resultName = ((ValidationWorkflowAware) action).getInputResultName();
} //获取执行的action方法上的@InputConfig注解
InputConfig annotation = action.getClass().getMethod(invocation.getProxy().getMethod(), EMPTY_CLASS_ARRAY).getAnnotation(InputConfig.class);
//如果确实加了这个注解的话
if (annotation != null) { if (!annotation.methodName().equals("")) {
//调用注解上配置的methodName指定的方法,执行这个方法,使用这个方法的返回值作为返回字串
Method method = action.getClass().getMethod(annotation.methodName());
//使用这个方法的返回值作为返回字串,这个方法可以做一些事情动态生成返回字串
resultName = (String) method.invoke(action);
} else {
//或者使用resultName作为返回字符串 ,这个值是直接指定的一个字符串
resultName = annotation.resultName();
}
} //悲剧了,拦截器栈不往下走了,到此结束掉(虽然在默认的拦截器栈里这就是最后一个了)
return resultName;
}
} //拦截器栈继续走
return invocation.invoke();
} }
字太多不看?呐呐,有图有真相:
总结一下:
这个拦截器是用来判断当前action的执行是否发生了错误以及发生错误时应该返回什么字串结果。
三、自定义返回结果字符串
返回的字串结果默认为Action.INPUT,允许自定义,自定义的办法有这么几种
1. 在配置拦截器的时候传入一个inputResultName的参数,这种是全局级别的,对所有的action都生效。
实现方式:修改配置文件
拷贝struts-default.xml的defaultStack拦截栈到我们自己的配置文件中,如下:
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE struts PUBLIC
"-//Apache Software Foundation//DTD Struts Configuration 2.0//EN"
"http://struts.apache.org/dtds/struts-2.0.dtd"> <struts> <package name="default" namespace="/" extends="struts-default"
abstract="false"> <interceptors>
<interceptor-stack name="defaultStack">
<interceptor-ref name="exception" />
<interceptor-ref name="alias" />
<interceptor-ref name="servletConfig" />
<interceptor-ref name="i18n" />
<interceptor-ref name="prepare" />
<interceptor-ref name="chain" />
<interceptor-ref name="debugging" />
<interceptor-ref name="scopedModelDriven" />
<interceptor-ref name="modelDriven" />
<interceptor-ref name="fileUpload" />
<interceptor-ref name="checkbox" />
<interceptor-ref name="multiselect" />
<interceptor-ref name="staticParams" />
<interceptor-ref name="actionMappingParams" /> <!-- 参数拦截器,就是这个拦截器将前台传过来的参数设置到ValueStack中去 -->
<interceptor-ref name="params">
<param name="excludeParams">dojo\..*,^struts\..*</param>
</interceptor-ref> <interceptor-ref name="conversionError" />
<interceptor-ref name="validation">
<param name="excludeMethods">input,back,cancel,browse</param>
</interceptor-ref> <!-- 处于拦截器栈的最后,必须在params拦截器后面调用,因为至少要把参数传进去才能知道到底有没有出错啊 :) -->
<interceptor-ref name="workflow">
<param name="excludeMethods">input,back,cancel,browse</param>
<param name="inputResultName">error</param>
</interceptor-ref>
</interceptor-stack>
</interceptors> <action name="loginAction" class="struts_practice_001.LoginAction">
<result>/success.jsp</result>
<result name="input">/login.jsp</result>
<result name="error">/error.jsp</result>
</action>
</package> </struts>
这个时候所有的action发现了fieldError或者actionError的时候返回的结果字串都是error了。
2. 让action实现ValidationWorkflowAware接口,然后可以返回一个字串结果,这种是action级别的,只对当前action有效。
实现方式:实现ValidationWorkflowAware接口:
ValidationWorkflowAware接口如下:
package com.opensymphony.xwork2.interceptor; /**
* <code>ValidationWorkflowAware</code>
*/
public interface ValidationWorkflowAware { String getInputResultName();
}
很明了的代码不用多解释,来action实现它:
public class LoginAction extends ActionSupport implements ValidationWorkflowAware { private String username;
private String passwd; @Override
public String execute() throws Exception {
return SUCCESS;
} @Override
public String getInputResultName() {
return ERROR;
}
//这个方法是在DefaultWorkflowInterceptor之前就执行了的,所以错误信息会被放入
@Override
public void validate() {
addFieldError("foo","bar");
} public String getUsername() {
return username;
} public void setUsername(String username) {
this.username = username;
} public String getPasswd() {
return passwd;
} public void setPasswd(String passwd) {
this.passwd = passwd;
} }
这就实现了action级别的控制,注意getInputResultName()是在DefaultWorkflowInterceptor执行的时候才会被调用的。
3. 给方法上加上@InputConfig注解,这种是方法级别的,粒度更细,仅对当前正在执行的方法有效。
实现方式:加@InputConfig注解
先看一下这个注解的代码:
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.METHOD})
public @interface InputConfig {
String methodName() default "";
String resultName() default Action.INPUT;
}
methodName是执行这个方法名的方法使用其返回值,这个可以在方法里面做一些事情动态生成结果字串之类的
resultName是直接使用传进来字符串作为结果字串
action代码:
public class LoginAction extends ActionSupport { private String username;
private String passwd; @InputConfig(resultName="addFail")
public String add(){
return SUCCESS;
} @InputConfig(resultName="deleteFail")
public String delete(){
return SUCCESS;
} //这个方法是在DefaultWorkflowInterceptor之前就执行了的,所以错误信息会被放入
@Override
public void validate() {
addFieldError("foo","bar");
} public String getUsername() {
return username;
} public void setUsername(String username) {
this.username = username;
} public String getPasswd() {
return passwd;
} public void setPasswd(String passwd) {
this.passwd = passwd;
} }
如此就实现了方法级别的粒度控制。
附录:
来纠正一个误区,就是如下的写法:
public class LoginAction extends ActionSupport { private String username;
private String passwd; @Override
public String execute() throws Exception {
System.out.println(username+":"+passwd); addFieldError("foo","bar");
addActionError("fooActionError");
return SUCCESS;
} public String getUsername() {
return username;
} public void setUsername(String username) {
this.username = username;
} public String getPasswd() {
return passwd;
} public void setPasswd(String passwd) {
this.passwd = passwd;
} }
这样写是没什么卵用的,为什么呢?因为要执行的方法是在拦截器栈全都执行完以后才会执行的,也就是说执行DefaultWorkflowInterceptor的时候execute()还没有被执行,所以正确的姿势是借助之前的拦截器的验证方法放入错误信息,比如下面这个就是可以的:
public class LoginAction extends ActionSupport { private String username;
private String passwd; @Override
public String execute() throws Exception {
System.out.println(username+":"+passwd); addFieldError("foo","bar");
addActionError("fooActionError"); return SUCCESS;
} //这个方法是在DefaultWorkflowInterceptor之前就执行了的,所以错误信息会被放入
@Override
public void validate() {
addFieldError("foo","bar");
}
public String getUsername() {
return username;
} public void setUsername(String username) {
this.username = username;
} public String getPasswd() {
return passwd;
} public void setPasswd(String passwd) {
this.passwd = passwd;
} }
至于这个方法为什么会先于execute()被调用,参考另一篇博文:(还木有写吶(=@__@=))
Struts2拦截器之DefaultWorkflowInterceptor的更多相关文章
- Struts2拦截器之FileUploadInterceptor
一.它能做什么? 借助于这个拦截器我们可以实现文件的上传和下载功能. 理论部分: struts2的文件上传下载功能也要依赖于Apache commons-fileupload和Apache commo ...
- Struts2拦截器之ModelDrivenInterceptor
叙述套路: 1.这是个啥东西,它是干嘛用的? 2.我知道它能干啥了,那它咋个用呢? 3.它能跑起来了,但是它是咋跑起来的是啥原理呢? 一.ModelDriven是个啥?他能做什么? 从前端页面到后端的 ...
- Struts2拦截器之ExceptionMappingInterceptor(异常映射拦截器)
一.异常拦截器是什么? 异常拦截器的作用是提供一个机会,可以设置在action执行过程中发生异常的时候映射到一个结果字符串而不是直接中断. 将异常整合到业务逻辑中,比如在分层系统的调用中可以从底层抛出 ...
- Struts2拦截器的执行过程浅析
在学习Struts2的过程中对拦截器和动作类的执行过程一度陷入误区,特别读了一下Struts2的源码,将自己的收获分享给正在困惑的童鞋... 开始先上图: 从Struts2的图可以看出当浏览器发出请求 ...
- Struts2拦截器总结<转>
由于项目中在登录跳转到其他应用程序模块的时候有用到拦截器,因此查看了一下相关资料. 原文地址:http://blog.csdn.net/sendfeng/article/details/4248120 ...
- Struts2 拦截器具体配置过程
拦截器差点儿遍布每个程序中,所以贴出拦截器配置的具体过程,希望可以帮到大家. Struts2 拦截器具体配置过程 <interceptors> <!-- 先定义拦截器 --> ...
- 笔记:Struts2 拦截器
配置拦截器 Struts.xml 配置文件中,使用<interceptor-/>来定义拦截器,有属性 name 表示拦截器的名称,class 表示拦截器的具体首先类,可以使用<par ...
- Struts2拦截器详解
一.Struts2拦截器原理: Struts2拦截器的实现原理相对简单,当请求struts2的action时,Struts 2会查找配置文件,并根据其配置实例化相对的 拦截器对象,然后串成一个列 ...
- Struts2拦截指定方法的拦截器
作者:禅楼望月 默认情况下,我们为一个Action配置一个拦截器,该拦截器会拦截该Action中的所有方法,但是有时候我们只想拦截指定的方法.为此,需要使用struts2拦截器的方法过滤特性. 要使用 ...
随机推荐
- ASP.NET、C#调用外部可执行exe文件--多种方案
一. try { //方法一 //调用自己的exe传递参数 //Process proc = new Process(); //proc.StartInfo.FileName = @"D:\ ...
- 【转】JavaScript面向对象
http://www.cnblogs.com/dolphinX/p/4385862.html 理解对象 对象这个词如雷贯耳,同样出名的一句话:XXX语言中一切皆为对象! 对象究竟是什么?什么叫面向对象 ...
- ASP.NET发送邮件(QQ发送)
public void SetEmail() { //电子邮件对象 MailMessage mailMessage = new MailMes ...
- Eclipse 语法提示
新建一个txt 拷贝下面的文本,然后保存修改扩展名为.epf #Sat Nov :: CST /instance/org.eclipse.jdt.core/org.eclipse.jdt.core.c ...
- python解释器快捷键
13. 交互式输入的编辑和历史记录 某些版本的 Python 解释器支持编辑当前的输入行和历史记录,类似于在 Korn shell 和 GNU Bash shell 中看到的功能.这是使用GNU Re ...
- buildroot 制作Linux文件系统初级使用教程
buildroot 下载地址:https://buildroot.org/download.html 放在Linux文件下解压出来. 使用make menuconfig 进行配置相关的东西. 在使用这 ...
- am335x UART1输入u-boot 调试信息代码修改
AM335x 调试信息UART1输出代码修改1. 关于pin_mux 的配置代码修改位置:/board/forlinx/ok335x/mux.c void enable_uart0_pin_mux( ...
- javascript,检测对象中是否存在某个属性
检测对象中属性的存在与否可以通过几种方法来判断. 1.使用in关键字. 该方法可以判断对象的自有属性和继承来的属性是否存在. var o={x:1}; "x" in o; //tr ...
- 关于linux中文乱码的问题。
公司让人在一台装有ubuntu14.04的机器上安装net-snmp,可是这台机器的设置很让人不喜.没关系,一个个解决它. 不能连接外网,得弄一个代理. 这个好说,在可以上外网的本机上安装squid工 ...
- ndk学习11: linux内存管理
1. 进程地址空间 2.内存管理 栈上分配空间 alloca() 栈上分配大小 strdupa() 拷贝一个字符串到栈上(显然这个函数不安全) ...