`
mj4d
  • 浏览: 304310 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

Struts2基于Annotation的服务端校验

阅读更多

在使用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

 

 

 

 

 

分享到:
评论

相关推荐

    Spring中文帮助文档

    2.5.3. 基于Annotation的控制器 2.5.4. Spring MVC的表单标签库 2.5.5. 对Tiles 2 支持 2.5.6. 对JSF 1.2支持 2.5.7. JAX-WS支持 2.6. 其他 2.6.1. 动态语言支持 2.6.2. 增强的测试支持 2.6.3. JMX 支持 ...

    Spring API

    2.5.3. 基于Annotation的控制器 2.5.4. Spring MVC的表单标签库 2.5.5. 对Tiles 2 支持 2.5.6. 对JSF 1.2支持 2.5.7. JAX-WS支持 2.6. 其他 2.6.1. 动态语言支持 2.6.2. 增强的测试支持 2.6.3. JMX 支持 ...

    基于MATLAB GUI与CNN的模糊车牌识别系统:从图像预处理到字符识别全流程解析

    内容概要:本文详细介绍了基于MATLAB GUI界面和卷积神经网络(CNN)的模糊车牌识别系统。该系统旨在解决现实中车牌因模糊不清导致识别困难的问题。文中阐述了整个流程的关键步骤,包括图像的模糊还原、灰度化、阈值化、边缘检测、孔洞填充、形态学操作、滤波操作、车牌定位、字符分割以及最终的字符识别。通过使用维纳滤波或最小二乘法约束滤波进行模糊还原,再利用CNN的强大特征提取能力完成字符分类。此外,还特别强调了MATLAB GUI界面的设计,使得用户能直观便捷地操作整个系统。 适合人群:对图像处理和深度学习感兴趣的科研人员、高校学生及从事相关领域的工程师。 使用场景及目标:适用于交通管理、智能停车场等领域,用于提升车牌识别的准确性和效率,特别是在面对模糊车牌时的表现。 其他说明:文中提供了部分关键代码片段作为参考,并对实验结果进行了详细的分析,展示了系统在不同环境下的表现情况及其潜在的应用前景。

    嵌入式八股文面试题库资料知识宝典-计算机专业试题.zip

    嵌入式八股文面试题库资料知识宝典-计算机专业试题.zip

    嵌入式八股文面试题库资料知识宝典-C and C++ normal interview_3.zip

    嵌入式八股文面试题库资料知识宝典-C and C++ normal interview_3.zip

    开关磁阻电机技术参数与建模技术深度解析:4kW电机性能详述

    内容概要:本文深入探讨了一款额定功率为4kW的开关磁阻电机,详细介绍了其性能参数如额定功率、转速、效率、输出转矩和脉动率等。同时,文章还展示了利用RMxprt、Maxwell 2D和3D模型对该电机进行仿真的方法和技术,通过外电路分析进一步研究其电气性能和动态响应特性。最后,文章提供了基于RMxprt模型的MATLAB仿真代码示例,帮助读者理解电机的工作原理及其性能特点。 适合人群:从事电机设计、工业自动化领域的工程师和技术人员,尤其是对开关磁阻电机感兴趣的科研工作者。 使用场景及目标:适用于希望深入了解开关磁阻电机特性和建模技术的研究人员,在新产品开发或现有产品改进时作为参考资料。 其他说明:文中提供的代码示例仅用于演示目的,实际操作时需根据所用软件的具体情况进行适当修改。

    少儿编程scratch项目源代码文件案例素材-剑客冲刺.zip

    少儿编程scratch项目源代码文件案例素材-剑客冲刺.zip

    少儿编程scratch项目源代码文件案例素材-几何冲刺 转瞬即逝.zip

    少儿编程scratch项目源代码文件案例素材-几何冲刺 转瞬即逝.zip

    四象限直流电机速度驱动控制系统PID控制仿真模型设计与实现

    内容概要:本文详细介绍了基于PID控制器的四象限直流电机速度驱动控制系统仿真模型及其永磁直流电机(PMDC)转速控制模型。首先阐述了PID控制器的工作原理,即通过对系统误差的比例、积分和微分运算来调整电机的驱动信号,从而实现转速的精确控制。接着讨论了如何利用PID控制器使有刷PMDC电机在四个象限中精确跟踪参考速度,并展示了仿真模型在应对快速负载扰动时的有效性和稳定性。最后,提供了Simulink仿真模型和详细的Word模型说明文档,帮助读者理解和调整PID控制器参数,以达到最佳控制效果。 适合人群:从事电力电子与电机控制领域的研究人员和技术人员,尤其是对四象限直流电机速度驱动控制系统感兴趣的读者。 使用场景及目标:适用于需要深入了解和掌握四象限直流电机速度驱动控制系统设计与实现的研究人员和技术人员。目标是在实际项目中能够运用PID控制器实现电机转速的精确控制,并提高系统的稳定性和抗干扰能力。 其他说明:文中引用了多篇相关领域的权威文献,确保了理论依据的可靠性和实用性。此外,提供的Simulink模型和Word文档有助于读者更好地理解和实践所介绍的内容。

    嵌入式八股文面试题库资料知识宝典-2013年海康威视校园招聘嵌入式开发笔试题.zip

    嵌入式八股文面试题库资料知识宝典-2013年海康威视校园招聘嵌入式开发笔试题.zip

    少儿编程scratch项目源代码文件案例素材-驾驶通关.zip

    少儿编程scratch项目源代码文件案例素材-驾驶通关.zip

    小区开放对周边道路通行能力影响的研究.pdf

    小区开放对周边道路通行能力影响的研究.pdf

    冷链物流路径优化:基于NSGA-2遗传算法与软硬时间窗策略的研究

    内容概要:本文探讨了冷链物流车辆路径优化问题,特别是如何通过NSGA-2遗传算法和软硬时间窗策略来实现高效、环保和高客户满意度的路径规划。文中介绍了冷链物流的特点及其重要性,提出了软时间窗概念,允许一定的配送时间弹性,同时考虑碳排放成本,以达到绿色物流的目的。此外,还讨论了如何将客户满意度作为路径优化的重要评价标准之一。最后,通过一段简化的Python代码展示了遗传算法的应用。 适合人群:从事物流管理、冷链物流运营的专业人士,以及对遗传算法和路径优化感兴趣的科研人员和技术开发者。 使用场景及目标:适用于冷链物流企业,旨在优化配送路线,降低运营成本,减少碳排放,提升客户满意度。目标是帮助企业实现绿色、高效的物流配送系统。 其他说明:文中提供的代码仅为示意,实际应用需根据具体情况调整参数设置和模型构建。

    少儿编程scratch项目源代码文件案例素材-恐怖矿井.zip

    少儿编程scratch项目源代码文件案例素材-恐怖矿井.zip

    基于STM32F030的无刷电机高压FOC控制方案:滑膜无感FOC技术及保护机制

    内容概要:本文详细介绍了基于STM32F030的无刷电机控制方案,重点在于高压FOC(磁场定向控制)技术和滑膜无感FOC的应用。该方案实现了过载、过欠压、堵转等多种保护机制,并提供了完整的源码、原理图和PCB设计。文中展示了关键代码片段,如滑膜观测器和电流环处理,以及保护机制的具体实现方法。此外,还提到了方案的移植要点和实际测试效果,确保系统的稳定性和高效性。 适合人群:嵌入式系统开发者、电机控制系统工程师、硬件工程师。 使用场景及目标:适用于需要高性能无刷电机控制的应用场景,如工业自动化设备、无人机、电动工具等。目标是提供一种成熟的、经过验证的无刷电机控制方案,帮助开发者快速实现并优化电机控制性能。 其他说明:提供的资料包括详细的原理图、PCB设计文件、源码及测试视频,方便开发者进行学习和应用。

    基于有限体积法Godunov格式的管道泄漏检测模型研究.pdf

    基于有限体积法Godunov格式的管道泄漏检测模型研究.pdf

    嵌入式八股文面试题库资料知识宝典-CC++笔试题-深圳有为(2019.2.28)1.zip

    嵌入式八股文面试题库资料知识宝典-CC++笔试题-深圳有为(2019.2.28)1.zip

    少儿编程scratch项目源代码文件案例素材-几何冲刺 V1.5.zip

    少儿编程scratch项目源代码文件案例素材-几何冲刺 V1.5.zip

    Android系统开发_Linux内核配置_USB-HID设备模拟_通过root权限将Android设备转换为全功能USB键盘的项目实现_该项目需要内核支持configFS文件系统.zip

    Android系统开发_Linux内核配置_USB-HID设备模拟_通过root权限将Android设备转换为全功能USB键盘的项目实现_该项目需要内核支持configFS文件系统

    C# WPF - LiveCharts Project

    C# WPF - LiveCharts Project

Global site tag (gtag.js) - Google Analytics