上面两篇文章文笔比较混乱,但能看出大概的过程。

 

个人实践:

1、概述:主要用到以下几个要素:一个与表单内容对应的JavaBean类,一个用来验证这个Bean的Validator类,一个调用这个Validator的Controller类。

所以,Validator需要知道JavaBean,Controller需要知道Validator。

剩下的都是细节。

2、例子:

(1) JavaBean



public class User {
    private Integer userId;

    private Integer parentId;

    private String userName;

    private String password;

    private String tel;

    private String email;

    private String qq;

    private String webChat;

    private Integer groupId;

    private Boolean state;

    private Date createTime;

    private Date lastTime;
}



getter和setter省略了。

 

(2)、表单jsp页面




<%@ page language="java" contentType="text/html; charset=UTF-8"
 pageEncoding="UTF-8"%>
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core"%>
<%@taglib uri="http://www.springframework.org/tags/form" prefix="form" %>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>添加用户</title>
<link href="/Public/media/css/new/page.css" rel="stylesheet"
 type="text/css" />
<style type="text/css">
.errormsg {
 font-family: "微软雅黑";
 color: #F30;
}
</style>
</head>
<body>
<form:form action="${pageContext.request.contextPath}/userConfig/addDA" method="post" commandName="user">
        <div class="formDiv">
            <table width="1020" border="0">
                <tr>
                    <td align="right">用户名:</td>
                    <td align="left"><form:input path="userName"/><span class="errormsg"><form:errors path="userName"/></span></td>
                </tr>
                <tr>
                    <td align="right">密码:</td>
                    <td align="left"><form:password path="password"/><span class="errormsg"><form:errors path="password"/></span></td>
                </tr>
                <tr>
                    <td align="right">电话:</td>
                    <td align="left"><form:input path="tel"/><span class="errormsg"><form:errors path="tel"/></span></td>
                </tr>
                <tr>
                    <td align="right">邮件:</td>
                    <td align="left"><form:input path="email"/><span class="errormsg"><form:errors path="email"/></span></td>
                </tr>
                <tr>
                    <td align="right">QQ:</td>
                    <td align="left"><form:input path="qq"/><span class="errormsg"><form:errors path="qq"/></span></td>
                </tr>
                <tr>
                    <td align="right">微信:</td>
                    <td align="left"><form:input path="webChat"/><span class="errormsg"><form:errors path="webChat"/></span></td>
                </tr>
                <tr>
                    <td align="right">状态:</td>
                    <td align="left"><table width="200">
                            <tr>
                                <td><label> <input type="radio"
                                        name="StatusRadioGroup" value="禁用" id="StatusRadioGroup_0" />
                                        禁用
                                </label></td>
                            </tr>
                            <tr>
                                <td><label> <input type="radio"
                                        name="StatusRadioGroup" value="生效" id="StatusRadioGroup_1" />
                                        生效
                                </label></td>
                            </tr>
                        </table></td>
                </tr>
                <tr>
                    <td></td>
                    <td><input name="" type="submit" value="提交" /></td>
                </tr>

            </table>
        </div>
        </form:form>

</body>
</html>



想要详细了解这里用到的springmvc form标签的意思,需要先学习相应知识:http://haohaoxuexi.iteye.com/blog/1807330

关键点:

a、commandName属性指定了后台Model中与表单对应的对象的名字,需要在后台modelAndView.setObject("user",user);如果不写该属性,默认key是"command",那在后台就要

    modelAndView.setObject("command",user)。

b、<form:input和<form:errors标签中使用path属性指定该表单项对应的user的属性名。

c、注意引入springmvc的form标签<%@taglib uri="http://www.springframework.org/tags/form" prefix="form" %>

 

(3)Validator代码




package com.leslie;
import org.springframework.validation.Errors;
import org.springframework.validation.ValidationUtils;
import org.springframework.validation.Validator;



import com.leslie.User;

public class UserValidator implements Validator{

    @Override
    public boolean supports(Class<?> clazz) {
        return User.class.equals(clazz);
    }

    @Override
    public void validate(Object obj, Errors e) {
//        ValidationUtils.rejectIfEmpty(e, "name", "name.empty");  
//        ValidationUtils.rejectIfEmpty(e, "gender", "null");  
//        ValidationUtils.rejectIfEmpty(e, "age", "null");  
//        User s = (User)obj;  
//        if(s.getAge() < 18){  
//            e.rejectValue("age","young");  
//        }else if(s.getAge() > 50){  
//            //e.reject("age", "old");  
//        }  
         
//        if(s.getName().length() < 1){ 
//            e.reject("name", "short"); 
//        }else if(s.getName().length() > 10){ 
//            e.reject("name", "long"); 
//        }
        ValidationUtils.rejectIfEmpty(e, "userName", null, "用户名不能为空");
        ValidationUtils.rejectIfEmpty(e, "password", null, "密码不能为空");
    }

}



关键点:

a、supports方法中指定了对哪个类进行验证。

b、ValidationUtils是springmvc的工具类,你可以方便地使用它进行常用的判断,你也可以直接使用e.rejectValue定义你自己的判断信息,

注释掉的语句列出了常用的用法。

c、rejectIfEmpty方法第三个参数是国际化properties文件中的key值,由于我没有使用国际化,此处使用了第四个参数,也就是默认值。

 

(4) Controller实现



@Controller
@RequestMapping("/userConfig")
public class SJUserConfigController {
@InitBinder
 public void initBinder(WebDataBinder binder) {
 binder.setValidator(new UserValidator());
 }@RequestMapping("/openAddUser")
    public ModelAndView openAddUser() {
        ModelAndView mv = new ModelAndView();
        User u = new User();
        mv.addObject("user",u);
        mv.setViewName("addUser");
        return mv;
    }

    @RequestMapping("/addUser")
    public ModelAndView addUser(@Valid @ModelAttribute User user, BindingResult br,
            HttpServletRequest requset) {
        ModelAndView mv = new ModelAndView();
        mv.addObject("user", user);
        if (br.hasErrors()) {
            mv.setViewName("addUser");
        } else {
            mv.setViewName("success");
        }
        return mv;
    }
}
@RequestMapping("/openAddUser")
    public ModelAndView openAddUser() {
        ModelAndView mv = new ModelAndView();
        User u = new User();
        mv.addObject("user",u);
        mv.setViewName("addUser");
        return mv;
    }

    @RequestMapping("/addUser")
    public ModelAndView addUser(@Valid @ModelAttribute User user, BindingResult br,
            HttpServletRequest requset) {
        ModelAndView mv = new ModelAndView();
        mv.addObject("user", user);
        if (br.hasErrors()) {
            mv.setViewName("addUser");
        } else {
            mv.setViewName("success");
        }
        return mv;
    }
}



关键点:

a、@InitBinder指定了Validator。

b、注意openAddUser这个方法,这个方法执行以后首次打开表单页供用户填写信息,mv.addObject("user",u);这句非常必要,u不能为null,否则jsp报错。

c、addUser方法处理表单提交上来的内容,@Valid指定了要对其修饰的对象进行验证,如果没有这个注解,springmvc就不会对表单进行验证。

 

3、后记

使用了上述的验证机制以后,虽然经过测试,可以正常验证表单,但后台服务器报

javax.validation.ValidationException: Unable to create a Configuration, because no Bean Validation provider could be found.

加入hibernate-validator.jar(以及其依赖的jboss-logging.jar和classmate.jar)以后就正常了,具体Jar的版本号此处略去。

 

4、后记的后记:一个巨大的坑和巨大的潜规则--commandName与@ModelAttribute必须配合默契否则不报错但就是不显示错误信息

什么意思呢?说明如下。

在前面的例子中,我很巧合地将页面的commandName值赋为"user",而我的表单对应Java类刚好也是叫User,而且我直接这样@ModelAttribute使用了注解,并没有指定@ModelAttribute("xxxx"),结果

一切顺利,但这只是一种巧合情况。

好,那下面我总结一下我实验出的最后结论:

1、如果你直接使用@ModelAttribute,并没有指定其value,那对不起,你页面中的commandName的值必须是你的表单对应类的类名的小写!最恶心的是,如果你不知道这一点,

随意地给commandName起名字的话,validator会正确验证表单,controller会正确跳转,一切都正常,就是页面中不显示错误信息!而且不报任何错误!

2、如果你这样使用@ModelAttribute注解,比如@ModelAttribute("abcdef"),那么你页面中的commandName也必须等于"abcdef",二者匹配起来用,这样,你才可能达到自己给commandName

起名字的目的!

 

后记的后记二:

如果一个controller中的方法处理两个及两个以上的form表单,比如,方法A()接收并处理用户注册信息,而方法B()接收并处理定单查询条件,由于controller只能绑定一个validator,没有与validator匹配的那个

表单在被提交之后也会被validator验证并报错,即使你没加@Valid也会。目前没有解决的办法,只能将这种情况拆开两个controller处理。