在使用Struts2开发时,经常会遇到在服务端Action方法中对数据有效性的校验(当然任何框架都会遇到),当遇到一大堆属性需要校验时就显得繁琐,而struts2本身的校验插件用起来也不是那么简单,最近自己就尝试用Annotation的方式对数据的有效性进行了校验。
首先简单介绍下验证思路:
1、制定校验的Annotaion,主要针对Field、方法级别
2、Annotation相应的校验规则
3、采用Struts2中的拦截器进行校验,在拦截器初始化方法中加载校验的Annotaion和校验规则
4、拦截器对请求方法和Action中的Field截取,读取Field上的Annotaion并采用3中保存的校验规则进行校验
5、将校验产生的不通过信息存储在ActionContext中,在具体的Action获取并处理消息
接下来看看具体的实现过程,将分为以下几部分:
1、Annotation和AnnotationChecker
这都是声明的常规性Annotation,主要作用在对象的Field上,以下是一个最大长度校验Annotation的声明:
@Target({ ElementType.FIELD })
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MaxLength {
public int value() default 0;
}
MaxLength的校验类MaxLengthChecker,这里约定检验类是在Annotation+Checker:
public class MaxLengthChecker implements Checker {
@Override
public CheckerResult check(Object value, String fieldName, Annotation annotation) {
if (null != value) {
try {
List<String> maxLeng = new ArrayList<String>();
MaxLength maxLength = (MaxLength) annotation;
String strVal = (String) value;
maxLeng.add(String.valueOf(maxLength.value()));
if (strVal.trim().length() > maxLength.value()) {
return new CheckerResult(fieldName, CheckerMsgConstant.CHECKER_MAXLENGTH_OVERFLOW, maxLeng);
}
} catch (Exception e) {
//转换为String失败或其他异常给出提示
return new CheckerResult(fieldName, CheckerMsgConstant.CHECKER_MAXLENGTH_NOTSTR);
}
}
return null;
}
}
这里所有Checker实现接口Checker,方法check作为Annotation的校验方法,返回CheckerResult,用来描述在哪个field和相应的失败规则,如下:
public class CheckerResult implements Serializable {
private static final long serialVersionUID = 103986447231347145L;
/** 受检查字段 */
private String field;
/** 检查结果错误类型 */
private String checkType;
/** 检查中传递的参数(如最大最小值等) */
private List<String> values;
2、Struts2拦截器初始化加载所有的校验Annotation
这是校验的主要方法,继承struts2中的拦截器,首先在初始化方法中根据指定路径加载Annotation和AnnotationChecker
/**
* load check rules
*/
public void init() {
try {
URL packageUrl = CheckerIntercepter.class.getClassLoader().getResource(
CKECHER_ANNOTATION_PACKAGE.replaceAll("\\.", "/"));
for (String file : new File(packageUrl.getFile()).list()) {
if (!file.endsWith(".class"))
continue;
String checkerName = file.substring(0, file.length() - 6);
Checker checker = (Checker) Class.forName(CKECHER_PACKAGE + "." + checkerName + "Checker")
.newInstance();
checkerMap.put(checkerName, checker);
}
} catch (Exception e) {
LOG.error(e.getMessage(), e);
}
}
这个会在容器启动时加载一次,接下来是拦截器的主要方法intercept,主要获取当前Action和执行的方法。
这里处理了几点:
1、如果方法有@Ignore就直接忽略
2、只处理当前请求参数的annotation
@Override
@SuppressWarnings("all")
public String intercept(ActionInvocation invocation) throws Exception {
Object obj = invocation.getAction();
//只处理当前方法的请求参数的校验
Set<String> paramSet = reqParams(invocation.getInvocationContext().getParameters());
if (null == paramSet || paramSet.size() == 0) {
return invocation.invoke();
}
reqParas = new WeakReference<Set<String>>(paramSet);
if (null != obj) {
//方式是否ignore
Method method = obj.getClass().getDeclaredMethod(invocation.getProxy().getMethod(), null);
boolean ignore = true;
//方法不能为空,并且忽略默认执行方法(execute)
if (null != method && !IGNORE_METHOD.equals(method)) {
//方法是否有Ignore标注
Annotation[] annotations = method.getAnnotations();
for (Annotation anno : annotations) {
if (IGNORE_ANNOTATION.equals(anno.annotationType().getSimpleName())) {
ignore = false;
break;
}
}
if (ignore) {
List<CheckerResult> list = populateObject(obj, null);
if (null != list && !list.isEmpty()) {
//如果有校验不通过的消息,存储在ActionContext中供各Action调用
invocation.getInvocationContext().put(CheckerMsgConstant.CHECKER, list);
}
}
}
}
return invocation.invoke();
}
3、对Action中的Field中的Annotation进行校验,并支持作为成员变量的自定义类中Field的校验。这里主要是上面的populateObject方法
/**
* 对当前对象(Action)或Action中的成员变量获取其Field以及相应的Annotation进行校验
*
* @param obj
* @param clazz
* @return
* @author robin
* @date 2012-7-18
*/
@SuppressWarnings("all")
private <T> List<CheckerResult> populateObject(Object obj, Class<T> clazz) {
List<CheckerResult> result = new ArrayList<CheckerResult>();
Class clz = obj == null ? clazz : obj.getClass();
//当前对象如果是Action获取成员变量,如果是作为成员变量的对象,获取相应的父类
List<Field> fields = new ArrayList<Field>();
while (!clz.equals(Object.class)) {
fields.addAll(Arrays.asList(clz.getDeclaredFields()));
if (clz.getSuperclass().equals(ActionSupport.class) || clz.equals(Class.class)) {
break;
}
clz = clz.getSuperclass();
}
for (Field field : fields) {
//是否有@Ignore、静态、接口、数组成员变量忽略
if (isIgnored(field)) {
continue;
}
//当前请求参数中是否有该Field
String fieldName = field.getName();
if (!reqParas.get().contains(fieldName)) {
continue;
}
Object object = null;
if (null != obj) {
try {
field.setAccessible(true);
object = field.get(obj);
} catch (Exception e) {
LOG.error("FAILED get field value", e);
//do nothing
}
}
if (isPrimitive(field.getType())) {
result.addAll(checker(field, object));
} else {
result.addAll(populateObject(object, field.getType()));
}
}
return result;
}
这里primitives是制定的默认直接进行校验的集合,不过这样做并不太合理。需要进一步考虑下,这是配置:
/**
* 判断是否是原生类型(如果是实体对象需要递归调用对其判断)
*
* @param clazz
* @return
* @author robin
* @date 2012-7-20
*/
private boolean isPrimitive(@SuppressWarnings("rawtypes") Class clazz) {
return clazz.isPrimitive() || clazz.equals(String.class) || clazz.equals(Date.class)
|| clazz.equals(Boolean.class) || clazz.equals(Byte.class) || clazz.equals(Character.class)
|| clazz.equals(Double.class) || clazz.equals(Float.class) || clazz.equals(Integer.class)
|| clazz.equals(Long.class) || clazz.equals(Short.class) || clazz.equals(Locale.class)
|| clazz.isEnum();
}
4、剩下的就是checker方法,对Field和该Field的值进行校验
/**
* 对当前Field的Annotation进行校验
*
* @param field
* @param value
* @return
* @author robin
* @date 2012-7-18
*/
private List<CheckerResult> checker(Field field, Object value) {
List<CheckerResult> result = new ArrayList<CheckerResult>();
Annotation[] annotations = field.getAnnotations();
for (Annotation anno : annotations) {
Checker checker = checkerMap.get(anno.annotationType().getSimpleName());
if (checker != null) {
CheckerResult check = checker.check(value, field.getName(), anno);
if (null != check) {
result.add(check);
}
}
}
return result;
}
最后来看看如何使用:
1、需要将当前的拦截器加到项目的default-interceptor-ref如:
<interceptors>
<interceptor name="checkerIntercepter" class="org.apache.struts2.valid.common.CheckerIntercepter" />
<interceptor-stack name="s2webDefaultStack">
<interceptor-ref name="defaultStack"/>
<interceptor-ref name="checkerIntercepter" />
</interceptor-stack>
</interceptors>
<default-interceptor-ref name="s2webDefaultStack"/>
2、为了方便使用,提供了一个ActionBase方法,可继承该Action,当然也可以自己处理,因为消息已经在struts2的数据上下文环境中:
List<CheckerResult> result = (List<CheckerResult>) ActionContext.getContext().get(CheckerMsgConstant.CHECKER);
这是ActionBase中获得消息的一个方法:
@SuppressWarnings("unchecked")
protected List<String> getValidInfos() {
List<String> resultList = new ArrayList<String>();
List<CheckerResult> result = (List<CheckerResult>) ActionContext.getContext().get(CheckerMsgConstant.CHECKER);
for (CheckerResult checkerResult : result) {
//注意:第一个参数默认为类型,为message中的key
//new String[]{/**默认为字段名称*/, ...(一个或多个参数)}
resultList.add(getText(checkerResult.getCheckType(), getValidateInfo(checkerResult)));
}
return resultList;
}
private String[] getValidateInfo(CheckerResult checker) {
List<String> values = checker.getValues();
String field = getText(checker.getField());
if (null != values && values.size() > 0) {
String[] result = new String[1 + values.size()];
result[0] = field;
int index = 1;
for (String rst : values) {
result[index] = rst;
index++;
}
return result;
}
return new String[] { field };
}
而验证的错误消息声明如下:
/** {0}长度不能超过{1} */
public static final String CHECKER_MAXLENGTH_OVERFLOW = "checker.maxlength.overflow";
/** {0}不是字符 */
public static final String CHECKER_MAXLENGTH_NOTSTR = "checker.maxlength.notstr";
这些消息的key已经在CheckerResult中的checkType体现,并且可以默认提供一系列的消息,如message.properties中:
checker.maxlength.overflow={0}\u957F\u5EA6\u4E0D\u80FD\u8D85\u8FC7{1}
3、在业务Action中,可以这样:
public class ValiAction extends ValidBaseAction {
private static final long serialVersionUID = -5951668645132875324L;
@Must
@MaxLength(5)
private String username;
@Must
private String password;
private PageModule<String> pageModule;
@Ignore
public String index() {
return SUCCESS;
}
public String save() {
//获得校验错误消息
getValidInfos();
以上可见:https://github.com/yooodooo/s2valid.git
分享到:
相关推荐
在这个小例子中,我们将深入探讨如何利用Struts2的Annotation配置来构建一个简单的应用。 首先,让我们了解什么是Annotation。在Java中,Annotation是一种元数据,它提供了一种安全的方式向编译器、JVM或者第三方...
在这个基于Annotation的小DEMO中,我们将探讨如何使用Struts2.3.15的注解特性来创建一个简单的Web应用。 首先,Struts2的注解简化了Action类的配置。在传统的Struts2配置中,我们通常会在struts.xml文件中定义每个...
Struts2是一个流行的Java web框架,它极大地简化了MVC(模型-...掌握这些注解的用法,能够帮助开发者更高效地开发基于Struts2的Java web应用程序。通过实践和不断探索,我们可以充分利用注解带来的便利,提升开发体验。
博文链接:https://flym.iteye.com/blog/174358
在给定的“struts2-Annotation”主题中,重点是Struts2框架如何利用注解(Annotation)来增强其功能和简化配置。注解是一种元数据,可以在代码中嵌入,提供有关类、方法或字段的额外信息,而无需编写XML配置文件。 ...
Struts2是一个强大的Java web应用程序框架,它基于MVC(Model-View-Controller)设计模式,为开发者提供了构建可维护性、可扩展性良好的Web应用的工具。在Struts2中,拦截器(Interceptor)是核心组件之一,它们在...
在Struts2中,使用注解(Annotation)可以简化配置,提高开发效率。本篇文章将深入探讨如何在Struts2中通过注解实现返回JSON数据的功能。 首先,让我们理解JSON(JavaScript Object Notation)是一种轻量级的数据...
在嵌入式Jetty环境下运行Struts2 Annotation项目是一个常见的任务,特别是在开发和测试阶段,因为这种方式能够快速启动服务,而无需依赖大型服务器容器。本文将深入探讨如何配置和执行这个过程,以及涉及的关键技术...
**二、基于Annotation的输入校验** Struts2提供了许多注解,位于`com.opensymphony.xwork2.validator.annotations`包下,可以直接在Action类的setter方法上使用这些注解进行校验。例如,使用`@NotEmpty`、`@Email`...
标题“struts2annotation json”暗示我们将探讨如何在Struts2中使用注解来处理JSON相关的功能。首先,让我们深入理解Struts2的注解系统。 1. **Struts2注解**: - `@Action`: 这个注解用于标记一个方法为处理HTTP...
Struts2是一个基于MVC架构的Java Web应用框架,它继承了Struts1的优点,并在此基础上进行了改进。Struts2框架的核心是拦截器机制,这使得开发者能够更加灵活地处理请求和响应。Struts2还集成了Spring和Hibernate等...
它支持两种验证方式:基于注解的验证(Annotation-based Validation)和基于XML的验证(XML-based Validation)。自定义校验主要是通过编写自定义校验器或扩展内置校验器来实现。 2. **自定义校验器实现** 自定义...
Struts2、Hibernate3和Spring2.5是Java Web开发中的三大框架,它们各自负责不同的职责,但可以协同工作以构建高效的企业级应用。这里主要讨论的是如何将这三者结合,并利用注解(Annotation)进行配置,以简化开发...
在Struts2框架中,使用注解(Annotation)可以极大地简化控制器类的配置,提高代码的可读性和维护性。本文将深入探讨如何利用Struts2的注解功能实现批量下载功能,并通过创建临时文件来处理下载请求,同时确保在下载...
Struts2是基于Java的一个开源MVC框架,它继承了Struts1的设计理念,并且采用了拦截器架构,这使得它可以更好地与Spring、Hibernate等其他框架集成。Struts2支持多种配置方式,如XML配置、注解配置等,其中注解配置因...