SpringMVC(二)

SpringMVC(二)

通过上一篇 SpringMVC 的博文,我们掌握了如何新建 SpringMVC 项目,了解了其大致工作原理,了解了常用的注解,知道了 REST 风格的架构,通过源码初步了解到了数据绑定的流程。接着上次我们继续对 SpringMVC 进行学习。

数据绑定、校验、格式化

  • SpringMVC 通过反射机制对目标处理方法进行解析,将请求消息绑定到处理方法的入参中。
  • 数据绑定流程
    1. SpringMVC 将 ServletRequest 对象及目标方法的入参实例传递给 WebDataBinderFactory 实例,以创建 WebDataBinder 实例
    2. DataBinder 调用装配在 SpringMVC 上下文中的 ConversionService 组件进行数据类型转换、数据格式化工作,将请求信息填充到入参对象中
    3. 调用 Validator 组件对已经绑定了请求信息的入参对象进行数据合法性校验,并最终生成数据绑定结果 BindingData 对象
    4. SpringMVC 抽取 BindingResult 中的入参对象和校验错误对象,将他们赋给处理方法的响应入参
  • **@InitBinder 注解**
    • 由 @InitBinder 标识的方法,可以对 WebDataBinder 对象进行初化。WebDataBinder 是 DataBinder 的子类,用于完成由表单字段 JavaBean 属性的绑定
    • @InitBinder方法不能有返回值,它必须声明为 void
    • @InitBinder方法的参数通常是 WebDataBinder,它可以对 DataBinder 进行初始化和一些设置,如设置绑定过程中使得某些字段不被赋值
  • 数据格式化
    • Spring 在格式化模块中定义了一个实现 ConversionService 接口的 FormattingConversionService 实现类,该实现类扩展了 GenericConversionService,因此它既具有类型转换的功能,又具有格式化的功能
    • 默认创建的 ConversionService 实例即为 FormattingConversionServiceFactroyBean(支持数据和日期的格式化)
    • FormattingConversionServiceFactroyBean 内部已经注册了 :
      • NumberFormatAnnotationFormatterFactroy:支持对数字类型的属性使用 **@NumberFormat** 注解
      • JodaDateTimeFormatAnnotationFormatterFactroy:支持对日期类型的属性使用 **@DateTimeFormat** 注解 @DateTimeFormat(pattern = "yyyy/mm/dd") private Date birth; @NumberFormat(pattern = "#,###,###.#") private float salary;
  • 数据校验
    • 如何校验
      • 使用 JSR303 验证标准
      • 加入 hibernate-validator 验证框架,即 jar 包
      • 加入
      • 在 bean 属性上添加对应的注解
      • 目标方法 bean 的属性上添加 @valid 注解
      • 注意:需要校验的 Bean 对象和其绑定结果或错误对象成对出现时,他们之间不允许声明其他入参
    • JSR 303
      • JSR 303 是 Java 为 Bean 数据合法性校验提供的标准框架,它已经包含在 JavaEE 6.0 中 .
      • JSR 303 通过在 Bean 属性上标注类似于 @NotNull、@Max 等标准的注解指定校验规则,并通过标准的验证接口对 Bean 进行验证
    • 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-filterinclude-filter 子节点规定只扫描的注解,SpringMVC 的 IOC 容器只扫描 **@Controller**(Handler 类) 和 **@ControllerAdvice**(处理异常的类),Spring 不扫描这两个注解注解的类
  • SpringMVC IOC 容器中的 bean 可以引用 Spring IOC 容器中的 Bean,反之不行。

大牛们,觉得有任何问题和不足还希望指出,共同进步,谢谢!

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏编程坑太多

『高级篇』docker之课程管理dubbo入门操练(14)

PS:dubbo的入门也就到这里,从spring 和springboot 对dubbo的整合。

12520
来自专栏web前端教室

【视频】Es6新特性-Symbol

温馨提示:视频请点此观看 视频原文: es6 中的symbol [ˈsɪmbl] symbol 是 ES6 的一个新特性 symbol 是一个 “新” 的...

21460
来自专栏木制robot技术杂谈

Python中os.path.dirname(__file__)的用法

os.path.dirname()的用途 os.path.dirname()用来获取文件的路径。 如: 1 2>>> os.path.dirname('/ho...

41770
来自专栏技术博文

file_put_contents— 将一个字符串写入文件

将字符串写入到文件中,我们可以用fwrite写文件函数进行操作,今天写程序的时候,突然觉得其实file_put_contents()函数,用来写入字符串,后来仔...

39770
来自专栏微信公众号:Java团长

学习SpringMVC——如何获取请求参数

  @RequestParam,你一定见过;@PathVariable,你肯定也知道;@QueryParam,你怎么会不晓得?!还有你熟悉的他(@CookieV...

9820
来自专栏互联网大杂烩

Spring MVC框架

前端控制器是DispatcherServlet;应用控制器其实拆为处理器映射器(Handler Mapping)进行处理器管理和视图解析器(View Resol...

7720
来自专栏我的博客

Zend_Config使用笔记

1.zend_Config被设计在应用程序中简化访问和使用配置数据。它为在应用程序代码中访问这样的配置数据提供了一个基于用户接口的嵌入式对象属性。配置数据可能来...

27450
来自专栏开发与安全

linux网络编程之socket(七):一个进程发起多个连接和gethostbyname等函数

一、在前面讲过的最简单的回射客户/服务器程序中,一个客户端即一个进程,只会发起一个连接,只要稍微修改一下就可以让一个客户端发起多个连接,然后只利用其中一个连接发...

27000
来自专栏阿杜的世界

Spring Boot:定制自己的starter

在学习Spring Boot的过程中,接触最多的就是starter。可以认为starter是一种服务——使得使用某个功能的开发者不需要关注各种依赖库的处理,不需...

12900
来自专栏JAVA同学会

Spring Data(一)概念和仓库的定义

Spring Data的主要任务是为数据访问提供一个相似的、一致的、基于Spring的编程模型,同时又保留着下面各个数据存储的特征。它使得使用数据访问技术非常的...

14110

扫码关注云+社区

领取腾讯云代金券