通过上一篇 SpringMVC 的博文,我们掌握了如何新建 SpringMVC 项目,了解了其大致工作原理,了解了常用的注解,知道了 REST 风格的架构,通过源码初步了解到了数据绑定的流程。接着上次我们继续对 SpringMVC 进行学习。
数据绑定、校验、格式化
-
SpringMVC 通过反射机制对目标处理方法进行解析,将请求消息绑定到处理方法的入参中。
-
数据绑定流程
-
SpringMVC 将 ServletRequest 对象及目标方法的入参实例传递给 WebDataBinderFactory 实例,以创建 WebDataBinder 实例
-
DataBinder 调用装配在 SpringMVC 上下文中的 ConversionService组件进行数据类型转换、数据格式化工作,将请求信息填充到入参对象中
-
调用 Validator 组件对已经绑定了请求信息的入参对象进行数据合法性校验,并最终生成数据绑定结果 BindingData 对象
-
SpringMVC 抽取 BindingResult 中的入参对象和校验错误对象,将他们赋给处理方法的响应入参
**@InitBinder 注解**
-
由 @InitBinder 标识的方法,可以对 WebDataBinder 对象进行初化。WebDataBinder 是 DataBinder 的子类,用于完成由表单字段 JavaBean 属性的绑定
-
@InitBinder方法不能有返回值,它必须声明为 void
-
@InitBinder方法的参数通常是 WebDataBinder,它可以对 DataBinder 进行初始化和一些设置,如设置绑定过程中使得某些字段不被赋值
数据格式化
-
NumberFormatAnnotationFormatterFactroy:支持对数字类型的属性使用 **@NumberFormat** 注解
-
JodaDateTimeFormatAnnotationFormatterFactroy:支持对日期类型的属性使用 **@DateTimeFormat** 注解
@DateTimeFormat(pattern = "yyyy/mm/dd")private Date birth;@NumberFormat(pattern = "#,###,###.#")private float salary;
-
Spring 在格式化模块中定义了一个实现 ConversionService 接口的 FormattingConversionService 实现类,该实现类扩展了 GenericConversionService,因此它既具有类型转换的功能,又具有格式化的功能
-
默认创建的 ConversionService 实例即为 FormattingConversionServiceFactroyBean(支持数据和日期的格式化)
-
FormattingConversionServiceFactroyBean 内部已经注册了 :
数据校验
-
JSR 303 是 Java 为 Bean 数据合法性校验提供的标准框架,它已经包含在 JavaEE 6.0 中 .
-
JSR 303 通过在 Bean 属性上标注类似于 @NotNull、@Max 等标准的注解指定校验规则,并通过标准的验证接口对 Bean 进行验证
-
使用 JSR303 验证标准
-
加入 hibernate-validator 验证框架,即 jar 包
-
加入
-
在 bean 属性上添加对应的注解
-
目标方法 bean 的属性上添加 @valid 注解
-
注意:需要校验的 Bean 对象和其绑定结果或错误对象成对出现时,他们之间不允许声明其他入参
-
如何校验
-
JSR 303
-
jar 包
<dependency> <groupId>org.hibernate</groupId> <artifactId>hibernate-validator</artifactId> <version>4.3.1.Final</version></dependency>
-
实体上添加注解
-
处理方法中添加 @valid 注解
格式化、校验错误的消息处理
-
若数据的校验以及格式化出错,我们先将其默认的错误消息打印到控制台,下面的代码将打印出具体的哪一个字段的什么错误。
@RequestMapping(value = "/emp", method = RequestMethod.POST)public String add(@Valid Employee employee, Errors result, Map<String, Object> map) {// 打印错误消息 if (result.getErrorCount() > 0) { System.out.println("出错了!"); for (FieldError fieldError : result.getFieldErrors()) { System.out.println(fieldError.getField() + " --> " + fieldError.getDefaultMessage()); } } employeeDao.save(employee); return "redirect:/employeeList"; }
-
将错误消息显示在页面上
-
转回原页面,并会回显输入的错误记录
-
页面上使用标签显示错误消息
-
-
如何覆盖错误消息以及将错误消息国际化
-
数据匹配是指是否和规定的格式一样,数据校验是指是否符合规定,我们可以配置国际化资源文件 i18n.properties,以及在 spring-c···onfig.xml 文件中配国际化资源,从而达到国际化资源消息的目标
-
properties 文件中声明不同错误对应不同的错误消息格式,对于校验使用对应的校验前缀(如 NotEmpty),对于类型错误使用 typeMismatch;
-
properties 文件中的第二个为 **@ModelAttribute** 标注的 value,或是类名的第一个字母消息,如 employee
-
SpringMVC 处理 JSON
-
导入 jar 包
<spring.verison>4.3.8.RELEASE</spring.verison> <jackson.version>2.8.7</jackson.version><dependency> <groupId>com.fasterxml.jackson.core</groupId> <artifactId>jackson-core</artifactId> <version>${jackson.version}</version></dependency><dependency> <groupId>com.fasterxml.jackson.core</groupId> <artifactId>jackson-annotations</artifactId> <version>${jackson.version}</version></dependency><dependency> <groupId>com.fasterxml.jackson.core</groupId> <artifactId>jackson-databind</artifactId> <version>${jackson.version}</version></dependency>
-
目标操作方法添加注解(目标方法返回一个对象或集合)
@ResponseBody @RequestMapping("/testJson") public Collection<Employee> testJson() { System.out.println("Succ"); Collection<Employee> values = employeeDao.getEmployeeMap().values();return values; }
-
目标页面发送 Ajax 请求以及处理返回值
$(function () { $("#testJson").click(function () { var url = this.href; var args = {}; $.post(url, args, function(data) { for (var i = 0; i < data.length; i++) { var lastName = data[i].lastName; var email = data[i].email; alert(lastName + ", " + email); } }) return false; }) })
SpringMVC 运行流程(其中
SpringMVC 和 Spring
-
需要进行 Spring 整合 SpringMVC 吗?
-
需要,通常情况下,将类似于数据源,事务,整合其他框架都是放在 Spring 的配置文件中,而不是 SpringMVC 文件中,实际上放入 Spring 配置文件对应的 IOC 容器中的还有 Service 和 Dao
-
不需要,都放在 SpringMVC 的配置文件中,也可以分多个 Spring 的配置文件,然后使用 import 节点导入其他的配置文件
-
-
问题
-
若 Spring 的 IOC 容器和 SpringMVC 的 IOC 容器扫描的包有重合的部分,就会导致 bean 被创建两次
-
-
解决
-
使用 exclude-filter 和 include-filter 子节点规定只扫描的注解,SpringMVC 的 IOC 容器只扫描 **@Controller**(Handler 类) 和 **@ControllerAdvice**(处理异常的类),Spring 不扫描这两个注解注解的类
-
-
SpringMVC IOC 容器中的 bean 可以引用 Spring IOC 容器中的 Bean,反之不行。
大牛们,觉得有任何问题和不足还希望指出,共同进步,谢谢!