Struts2之Action
MVC模式中需要有一个控制器来负责浏览器与服务器之间的通信,实现用户与服务器的交互。在Struts2框架中实现这一功能的是Action,它是整个框架最核心的部分。Action负责存储数据和状态,与应用的具体业务逻辑打交道。
一、Action简介
实现Struts2框架的Action有两种方案:
- 实现框架所提供的Action接口
- 使用框架所提供的ActionSupport基类
无论使用哪一种都必须实现execute()方法,返回的则是逻辑视图名。
二、返回结果
Action执行完了之后的返回类型为字符串,也就是逻辑视图名,根据它才能控制页面跳转,所在位置为com.opensyphony.xwork2.action中,就是框架所定义的接口。
com.opensyphony.xwork2.action
package com.opensymphony.xwork2;public interface Action { public static final String SUCCESS = "success"; public static final String NONE = "none"; public static final String ERROR = "error"; public static final String INPUT = "input"; public static final String LOGIN = "login"; public String execute() throws Exception;}
注意:对于返回类型的含义没有强制性要求,用户可以在控制器执行成功的情况下返回SUCCESS,并为之配置出错的返回页面,系统不会报错,但是出于对代码可读性和可维护性的考虑,还是按照返回类型的字面含义去配置它,例如Action正确执行则返回SUCCESS,出错则返回ERROR,需要返回到输入界面则返回INPUT等等。
Action中用到的是大写的返回类型名,而在配置文件struts.xml中则对应的是小写的字符串值,配置文件将控制器Action,返回结果类型以及物理视图对应起来。
三、execute()方法
Struts2中所有的Action都要实现execute()方法。用户的业务逻辑就是在该方法中处理,然后通过返回类型对到对应的视图,Action的作用就相当于Struts2改进的MVC模式的派发器作用。
实现execute()方法有两种方法,一个是实现com.opensymphony.xwork2.action接口,还有一种是从com.opensyphony.xwork2.ActionSupport基类派生。
方法一:实现Action接口
public class helloworld implements Action { public String execute() { return logicView; }}
可以看出实现Action的接口要求很简单,同时分析上下文给出的Action源文件可以发现功能有限,没有给出开发者所需的额外功能,因此这样的实现方法比较适合简单的Action,对于稍复杂的应用应该考虑使用继承ActionSupport基类。
方法二:继承ActionSupport基类
public class helloworld extends ActionSupport { public String execute() { return logicView; }}
表面上这两个Action没有太大的差别,都是实现了execute方法,然后返回一个字符串类型的逻辑视图。ActionSupport除了提供近30的成员函数涵盖了国际化、校验、出错处理等各个方面来处理用户的业务逻辑。所以如果用户的业务逻辑不是足够简单的话,就应该选择从ActionSupport基类中派生出自己的Action。
P.S.
一般来说接口要比较常用些,这次看到一个继承比接口要牛叉的了。ActionSupport基类中有很多很有用的成员函数,这里就不一一列出来了,但不是所有的都会经常使用,一般常用的都是些出错处理函数,如hasErrors()、hasFieldErrors()等等。
ActionSupport源代码来自:com.opensyphony.xwork2.ActionSupport
package com.opensymphony.xwork2;public class ActionSupport implements Action,Validateable,ValidationAware,TextProvider,LocaleProvider,Serializable,ContinuableObject { protected transient static final Log LOG = LogFactory.getLog(ActionSupport.class); private transient final TextProvider textProvider = new TextProviderSupport(getClass(),, this); private final ValidationAwareSupport validationAware = new ValidationAwareSupport(); // 设置 Action 级别的所有错误 public void setActionErrors(Collection errorMessages) { validationAware.setActionErrors(errorMessages); } // 获得 Action 级别的错误集合 public Collection getActionErrors() { return validationAware.getActionErrors(); } /*省略语句*/ public Map getErrors() { return getFieldErrors(); } /*省略语句*/ public Locale getLocale() { return ActionContext.getContext().getLocale(); } // 基于键获得指定的键值,不存在则返回 NULL public String getText(String aTextName) { return textProvider.getText(aTextName); } // 向 Action 添加一个 Action 级别的错误 public void addActionError(String anErrorMessage) { validationAware.addActionError(anErrorMessage); } /*省略语句*/ public String doInput() throws Exception { return INPUT; } // 是否有 Action 级别的错误 public boolean hasActionErrors() { return validationAware.hasActionErrors(); } // 清除所有的错误和信息 public void clearErrorsAndMessages() { validationAware.clearErrorsAndMessages(); } // 默认的空校验器 public void validate() { } // 暂停 Action 的执行,抛出异常,并转向对应返回类型 public void pause(String result) { }}
果然ActionSupport类的内容还是挺丰富的。
ValidationAware这个类对于ActionSupport类是一个非常重要的类,所有的Error,Message和FieldError都间接存放在ValidationAware这个类中。
validationAware类源码:com.opensyphony.xwork2.validationAware
public class ValidationAwareSupport implements ValidationAware, Serializable { private Collection actionErrors; private Collection actionMessages; private Map fieldErrors; /*省略语句*/}
由此可以看出,Error,Message是存放在Collection中,而FieldError存放在Map结构中。所以Action对这三个变量的操作其实是对这三种数据结构的操作。
四、Action实例
- 实现了Action接口,实现了execute方法 用来改变jsp中文本颜色
- 继承了ActionSupport基类,并重写了execute()方法 判断输入的参数是不是质数
五、Action中的校验初步
- 在Action的execute()方法中手动实现
- 使用ActionSupport类中的validate()方法 与基本的Action相比,ActionSupport实现了很多的成员函数,特别是为校验提供了很多的成员函数,而且ActionSupport更是实现了validate()方法,将检验与业务逻辑的流程独立开来。
- 使用注解校验
六、小结
- 阐述了 Struts2 中最基础、但同时也是最核心的Action,分别详细讲解了以实现Action接口和继承ActionSupport基类去实现一个Action,并对这两种方法的优缺点进行了比较。
- 通过分别通过两个例子验证这两种实现Action接口的方法,以期加深读者对这两者方式的印象。
- 接着介绍了简单的Action校验,校验是应用程序不可或缺的部分,而注释校验带来的便利是肯定的。
- 熟练掌握Action的编写是使用Struts2框架的基础,也是比较有技巧的一部分。
附:使用Struts2的注解校验
如果程序员不想自己写校验代码,也可以使用Struts2提供的校验功能代码。
同时Struts提供了注解来实现校验。Struts2本身提供了许多校验器供供编程人员使用,这些校验器可以方便的校验像输入是否为空、是否符合特定的格式(比如Date格式、email格式)等规则。
Struts2每一种校验都有特定的类来实现。
编程过程中只要导入包com.opensymphony.xwork2.validator.annotations,就可以使用这些校验器。要使用注解实现校验,可以按照一下的步骤
- 创建一个类或者接口,一般创建继承com.opensymphony.xwork2.ActionSupport的类。
- 为类创建字段变量,并为这些变量创建getter和setter方法
- 使用com.opensymphony.xwork2.annotations.Validation注解上面的Action类 告诉Struts2将为该类使用注解校验。
- 为setter()方法或execute()方法添加校验器。这里是实现对不同字段使用不同校验器的地方。注意使用哪个注解校验器就要导入对应校验器
例一,声明要使用注解了:
@Validation()public calss AnnotationAction extends ActionSupport { }
例二,对setter()方法进行注解:
@IntRangeFieldValidator(type=ValidatorType.FIELD,min="6",max="10",message="age must between ${min} and ${max}, current value is ${bar}.")public void setAge(int age) { this.age = age;}
例三,对execute()方法中的表达式进行注解:
expression = { @ExpressionValidator(expression = "password == confirm", message = "password must equal to confirm. password=${password},confirm=${confirm}.})})public String execute() throws Exception {}
做了这个例子之后还是有错误:
- 把该写在jsp上写到了s if中去了,结果页面中啥都没有
- 另外field这个单词经常拼写成filed
- 验证功能还是挺强大的,利用现有资源可以少写很多代码