struts2校验有两种实现方法:
    1. 手工编写代码实现(基本验证)
    2. 基于XML配置方式实现(验证框架)
 
I:首先 创建一个EmployeeAction类继承于ActionSupport 
对应的页面表单元素就一个empId 
 
public class EmployeeAction extends ActionSupport {
 
     private Integer empId;
     public void setEmpId(Integer empId) {
          System. out.println("setEmpId...." );
           this.empId = empId;
     } 
     public String save() {
          System. out.println("save....." );
           employeeService.save(empId );
           return "save-success" ;
     }
     //手动验证 这个方法是从ActionSupport 中继承得到的 因为ActionSupport 类实现          了
Validateable的验证接口 
    //public class ActionSupport implements Action, Validateable, ValidationAware
 
     @Override
     public void validate() {
           //这个方法是实现ValidationAware接口得到 给empId 属性手动添加一个错误消息 然后返回到浏览器进行显示
           this.addFieldError("empId" , "empId为空值" );
            
          System. out.println("validate....." );
     }  
}
 
基本原理:
 
 I:  this .addFieldError("empId" , "empId为空值" );
//ActionSupport 类中
 II: public void addFieldError (String fieldName, String errorMessage) {
        validationAware.addFieldError(fieldName, errorMessage);
  }
在ValidationAware的实现类中定义了这个方法接口的内容
III:
public class ValidationAwareSupport implements ValidationAware, Serializable {
 
      private Collection<String> actionErrors ;
      private Collection<String> actionMessages ;
      private Map<String, List<String>> fieldErrors ;
 
  
   public synchronized void addFieldError (String fieldName, String errorMessage) {
 
        Map<String, List<String>> errors = internalGetFieldErrors();获取一个Map集合
          //根据fieldName获取一个List集合
        List<String> thisFieldErrors = errors.get(fieldName);
 
        if (thisFieldErrors == null) {//判断集合是否为空 为空就创建一个List集合
            thisFieldErrors = new ArrayList<String>();
            errors.put(fieldName, thisFieldErrors);
        }
 
        thisFieldErrors.add(errorMessage);//把fieldName和错误消息 放入集合中
    }
     这个方法说明最终错误消息添加到fieldErrors Map集合中
     private Map<String, List<String>> internalGetFieldErrors() {
        if (fieldErrors == null) {
            fieldErrors = new LinkedHashMap<String, List<String>>();
        }
 
        return fieldErrors ;
    }
 
}
 
显示在debug中可以看到
 
总结:
Action中
    *  要继承ActionSupport
    *  重写Validateable接口中的validate()方法 ,在该方法中完成验证
           * 步骤如下:
                 * validate()方法在其他的业务方法之前执行
                 * 验证出错转向的页面
                         struts.xml配置<result name="input">/validate/XXX.jsp</result>  
                         其中input转向是在action中已经定义好的.
                         public static final String INPUT = "input";
                 
                 *   什么时候表示验证出错(转向input所指向的页面)
                         *  this.addFieldError("sss", "错误信息");方法指向的是一个集合
                         *  当集合不为空时,转向错误页面.
 
显示错误Jsp页面:
      使用<s:fielderror/>显示错误消息
 
补充
    validate()方法会校验action中所有与execute方法签名相同的方法。 
*  要校验指定的方法通过重写validateXxx()方法实现, validateXxx()只会校验action中
   方法名为Xxx的方法。其中Xxx的第一个字母要大写。
*  当某个数据校验失败时,调用addFieldError()方法往系统的fieldErrors添加校验失败
   信息(为了使用addFieldError()方法,action可以继承ActionSupport ),如果系统
   的fieldErrors包含失败信息,struts2会将请求转发到名为input的result。
*  在input视图中可以通过<s:fielderror/>显示失败信息。
          validateXxx()方法使用例子:
 
public String add() throws Exception{
     return "success";
}
 
对应的验证方法    
public void validateAdd(){
          //验证代码
}
在ValidationInterceptor拦截器中可以看到以validate开始的方法
 
 
源码中可以看到:
 
public class ValidationInterceptor extends MethodFilterInterceptor {
 
    private final static String VALIDATE_PREFIX = "validate";
    private final static String ALT_VALIDATE_PREFIX = "validateDo";
    
      
    protected void doBeforeInvocation(ActionInvocation invocation) throws Exception {
        Object action = invocation.getAction();
        ActionProxy proxy = invocation.getProxy();
 
        String context = this.getValidationContext(proxy);
        String method = proxy.getMethod();
  
        if (declarative ) {
           if (validateAnnotatedMethodOnly ) {
               actionValidatorManager.validate(action, context, method);
           } else {
               actionValidatorManager.validate(action, context);
           }
       }    
        if (action instanceof Validateable && programmatic) {
          
            Exception exception = null;
           
            Validateable validateable = (Validateable) action;
             
            try {
                    //这个和Preparable 拦截器原理相同
                  //在这里获取带有方法前缀的的方法 进行调用
             //  VALIDATE_PREFIX = "validate" ; 
            //ALT_VALIDATE_PREFIX = "validateDo" ;
                PrefixMethodInvocationUtil.invokePrefixMethod(
                                invocation,
                                new String[] { VALIDATE_PREFIX , ALT_VALIDATE_PREFIX });
            }
            catch(Exception e) { 
                exception = e;
            }
            
        }
    }
    @Override
    protected String doIntercept(ActionInvocation invocation) throws Exception {
        doBeforeInvocation(invocation); 
        return invocation.invoke();
    } 
}
public class PrefixMethodInvocationUtil {
 
public static void invokePrefixMethod (ActionInvocation actionInvocation, String[] prefixes) {
          Object action = actionInvocation.getAction(); 
          String methodName = actionInvocation.getProxy().getMethod(); 
           if (methodName == null) { 
             methodName = DEFAULT_INVOCATION_METHODNAME;
          } 
          //获取带前缀的方法
          Method method = getPrefixedMethod(prefixes, methodName, action);
           if (method != null) {
               //调用对应的action方法
              method.invoke(action, new Object[0]);
          }
     }
}
//获取带前缀的方法
public class PrefixMethodInvocationUtil {
     
public static Method getPrefixedMethod (String[] prefixes, String methodName, Object action) {
           assert(prefixes != null);
          String capitalizedMethodName = capitalizeMethodName(methodName);
        for (String prefixe : prefixes) {
            String prefixedMethodName = prefixe + capitalizedMethodName;
            try {
                return action.getClass().getMethod(prefixedMethodName, EMPTY_CLASS_ARRAY);
            }
            catch (NoSuchMethodException e) {
                 e
            }
        }
           return null ;
     }
}
 
输入校验的流程:
 
1。类型转换器对请求参数执行类型转换,并把转换后的值赋给action中的属性。
2。如果在执行类型转换的过程中出现异常,系统会将异常信息保存到ActionContext,conversionError拦截器将异常信息封装到fieldErrors里,然后执行第3步。如果类型转换没有出现异常,则直接进入第3步。
3。系统通过反射技术调用action中的validateXxx()方法,Xxx为方法名。
4。调用action中的validate()方法。
5。经过上面4步,如果系统中的fieldErrors存在错误信息(即存放错误信息的集合的size大于0),系统自动将请求转发至名称为input的视图。如果系统中的fieldErrors没有任何错误信息,系统将执行action中的处理方法。