Spring MVC 使用介绍(八)—— 类型转换
一、概述
spring类型转换有两种方式:
- PropertyEditor:可实现String<--->Object 之间相互转换
- Converter:可实现任意类型的相互转换
类型转换的过程中,当两者同时存在时,spring首先查找PropertyEditor进行类型转换,如果没有找到,则再查找Converter进行转换
二、PropertyEditor
1、基本介绍
PropertyEditor用于 String<--->Object 之间相互转换,spring内建了一些常用的PropertyEditor,如:
ClassEditor: String<——>Class
FileEditor: String<——>File
PatternEditor: String<——>Pattern
URLEditor: String<——>URL
ResourceEditor: String<——>Resource
自定义的PropertyEditor须继承自PropertyEditorSupport,示例如下:
实体类
public class UserInfo {
private String name;
private Integer age;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Integer getAge() {
return age;
}
public void setAge(Integer age) {
this.age = age;
}
}
转换类
public class UserEditor extends PropertyEditorSupport {
public UserEditor() {
System.out.println("UserEditor constructed");
}
@Override
public String getAsText() {
UserInfo userInfo = (UserInfo) this.getValue();
return JSON.toJSONString(userInfo);
}
@Override
public void setAsText(String text) throws java.lang.IllegalArgumentException {
if (StringUtils.isEmpty(text)) {
return;
}
UserInfo userInfo = JSON.parseObject(text, UserInfo.class);
this.setValue(userInfo);
}
}
控制器
@Controller
public class TestController5 { @RequestMapping("/type1")
@ResponseBody
public String testType1(@RequestParam("user") UserInfo userInfo) {
System.out.println(userInfo.getName() + " " + userInfo.getAge());
return "testType1";
} @RequestMapping("/type2")
@ResponseBody
public String testType2() {
System.out.println("void");
return "testType2";
}
// 注册UserEditor
@InitBinder
public void initBinder(WebDataBinder binder) {
UserEditor userEditor = new UserEditor();
binder.registerCustomEditor(UserInfo.class, userEditor);
System.out.println("initBinder invoked");
}
}
web.xml与spring-mvc.xml配置略
启动后,先后访问:
http://localhost:8080/myweb/type1?user={"name":"matt","age":30}
http://localhost:8080/myweb/type2
http://localhost:8080/myweb/type1?user={"name":"matt","age":30}
输出:
UserEditor constructed
initBinder invoked
matt 30
void
UserEditor constructed
initBinder invoked
matt 30
从输出结果可以看出,处理方法若包含需要类型转换的参数,每次请求都会调用注册方法;处理方法如不包含则不会调用
2、自定义PropertyEditor的注册
自定义PropertyEditor的注册有三种级别:
- 控制器级别,使用@InitBinder
- web级别,使用WebBindingInitializer
- 应用级别,自定义PropertyEditor与实体类同包,且命名为“实体名 + Editor”
i)控制器级别
控制器级别是在控制器类中使用@InitBinder注解来实现,注册范围为单个控制器,如上例所示
ii)web级别
web级别使用WebBindingInitializer来实现,注册范围为应用的所有控制器(即整个web层),示例:
WebBindingInitializer实现类
public class MyWebBindingInitializer implements WebBindingInitializer {
public void initBinder(WebDataBinder binder, WebRequest request) {
binder.registerCustomEditor(UserInfo.class, new UserEditor());
System.out.println("initBinder invoked!");
}
}
注册
<bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter">
<property name="webBindingInitializer">
<bean class="cn.matt.convertor.MyWebBindingInitializer"/>
</property>
</bean>
注释上例控制器中的注册方法,即可测试
该注册方法的调用方式与控制器级别相同,即处理方法若包含需要类型转换的参数,每次请求都会调用注册方法;处理方法如不包含则不会调用
补充:web级别的注册方法还有:1、基于继承的BaseController,2、基于@ControllerAdvice注解,这两种方式更简洁,详细可参考:
SpringMVC中利用@InitBinder来对页面数据进行解析绑定
Spring Boot 系列(八)@ControllerAdvice 拦截异常并统一处理
iii)应用级别(推荐)
应用级别的注册范围为spring层,使用方式:自定义PropertyEditor与实体类同包,且命名为“实体名 + Editor”,示例:UserInfoEditor
该注册方法在每次类型转换时均创建新的实例
三、Converter
1、基本介绍
Converter接口定义:
public interface Converter<S, T> {
T convert(S source);
}
Converter作为类型转换器,用于转换S类型到T类型,其实现必须是线程安全的且可以被共享,内建的常用Converter:
StringToBooleanConverter: String----->Boolean
StringToEnumConverterFactory: String----->enum类型
EnumToStringConverter: enum类型----->String
ConverterRegistry、ConversionService分别用于Converter的注册和运行支持,接口定义如下:
public interface ConverterRegistry {
void addConverter(Converter<?, ?> converter);
void addConverter(Class<?> sourceType, Class<?> targetType, Converter<?, ?> converter);
void addConverter(GenericConverter converter);
void addConverterFactory(ConverterFactory<?, ?> converterFactory);
void removeConvertible(Class<?> sourceType, Class<?> targetType);
}
public interface ConversionService {
boolean canConvert(Class<?> sourceType, Class<?> targetType);
boolean canConvert(TypeDescriptor sourceType, TypeDescriptor targetType);
<T> T convert(Object source, Class<T> targetType);
Object convert(Object source, TypeDescriptor sourceType, TypeDescriptor targetType);
}
Spring提供了两个默认实现(其都实现了ConverterRegistry、ConversionService接口):
- DefaultConversionService:默认的类型转换服务实现
- DefaultFormattingConversionService:带数据格式化支持的类型转换服务实现,一般使用该实现即可
使用示例如下:
转换器
public class StringToUserInfoConverter implements Converter<String, UserInfo> {
public StringToUserInfoConverter() {
System.out.println("StringToUserInfoConverter constructed");
}
public UserInfo convert(String source) {
if (StringUtils.isEmpty(source)) {
return null;
}
UserInfo userInfo = JSON.parseObject(source, UserInfo.class);
return userInfo;
}
}
测试
@Test
public void test() {
DefaultFormattingConversionService conversionService = new DefaultFormattingConversionService();
conversionService.addConverter(new StringToUserInfoConverter());
UserInfo userInfo = conversionService.convert("{\"name\":\"matt\",\"age\":30}", UserInfo.class);
System.out.println(userInfo.getName());
}
2、spring mvc中使用Converter
配置
<bean id="conversionService" class="org.springframework.format.support.FormattingConversionServiceFactoryBean">
<property name="converters">
<list>
<bean class="cn.matt.convertor.StringToUserInfoConverter"/>
</list>
</property>
</bean> <bean id="webBindingInitializer" class="org.springframework.web.bind.support.ConfigurableWebBindingInitializer">
<property name="conversionService" ref="conversionService"/>
</bean> <bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter">
<property name="webBindingInitializer" ref="webBindingInitializer" />
</bean>
控制器
@Controller
public class TestController6 {
@RequestMapping("/converter")
@ResponseBody
public String testConverter(@RequestParam("user") UserInfo userInfo) {
System.out.println(userInfo.getName() + " " + userInfo.getAge());
return "testConverter";
}
}
启动后,访问http://localhost:8080/myweb/type1?user={"name":"matt","age":30} 即可
补充:当使用<mvc:annotation-driven />时,配置如下
<mvc:annotation-driven conversion-service="conversionService"/> <bean id="conversionService" class="org.springframework.format.support.FormattingConversionServiceFactoryBean">
<property name="converters">
<list>
<bean class="cn.matt.convertor.StringToUserInfoConverter"/>
</list>
</property>
</bean>
参考:
SpringMVC数据类型转换——跟着开涛学SpringMVC
Spring MVC 使用介绍(八)—— 类型转换的更多相关文章
- Spring MVC 使用介绍(十三)数据验证 (一)基本介绍
一.消息处理功能 Spring提供MessageSource接口用于提供消息处理功能: public interface MessageSource { String getMessage(Strin ...
- Spring MVC 使用介绍(十五)数据验证 (二)依赖注入与方法级别验证
一.概述 JSR-349 (Bean Validation 1.1)对数据验证进一步进行的规范,主要内容如下: 1.依赖注入验证 2.方法级别验证 二.依赖注入验证 spring提供BeanValid ...
- Spring MVC 使用介绍(十四)文件上传下载
一.概述 文件上传时,http请求头Content-Type须为multipart/form-data,有两种实现方式: 1.基于FormData对象,该方式简单灵活 2.基于<form> ...
- 2017.3.31 spring mvc教程(八) <mvc:annotation-driven />所做的工作
学习的博客:http://elf8848.iteye.com/blog/875830/ 我项目中所用的版本:4.2.0.博客的时间比较早,11年的,学习的是Spring3 MVC.不知道版本上有没有变 ...
- Spring MVC JSON自己定义类型转换(续)
前面提到了两种转换类型的方法(Spring MVC JSON自己定义类型转换),这里针对Json转换提供一种更简便的方法. 通过配置全局的日期转换来避免使用麻烦的注解. 首先用到了一个简单的日期工具类 ...
- spring mvc绑定参数之 类型转换 有三种方式:
spring mvc绑定参数之类型转换有三种方式: 1.实体类中加日期格式化注解(上次做项目使用的这种.简单,但有缺点,是一种局部的处理方式,只能在本实体类中使用.方法三是全局的.) @DateTim ...
- Spring MVC 使用介绍(五)—— 注解式控制器(一):基本介绍
一.hello world 相对于基于Controller接口的方式,基于注解方式的配置步骤如下: HandlerMapping 与HandlerAdapter 分别配置为RequestMapping ...
- Spring MVC 使用介绍(十二)控制器返回结果统一处理
一.概述 在为前端提供http接口时,通常返回的数据需要统一的json格式,如包含错误码和错误信息等字段. 该功能的实现有四种可能的方式: AOP 利用环绕通知,对包含@RequestMapping注 ...
- spring mvc简单介绍xml版
spring mvc介绍:其实spring mvc就是基于servlet实现的,只不过他讲请求处理的流程分配的更细致而已. spring mvc核心理念的4个组件: 1.DispatcherServl ...
随机推荐
- Linux系统使用
linux(操作系统的内核) 浏览器功能:(内核的解释) 各个浏览器 实现的方式不一样 呈现内容 //解析内容和样式 用—webkit— (内核)解析 实现交互逻辑 v8 引擎 (内核) 实现 =&g ...
- AEAI CRM V1.6.0 升级说明,开源客户关系管理系统
1 升级说明 AEAI CRM v1.6.0版是AEAI CRM v1.5.2版客户关系管理系统的升级版本,本次版本是基于AEAI DP v3.8.0_20170228进行打包部署的,升级内容主要是针 ...
- Flask框架搭建REST-API服务
一.目的 为了能够将测试工具部署成RESTful-API服务,这样就能通过接口的方式提供统一测试工具服务,使用人员就不用构建application而产生的各种环境问题.使用问题. 适合人群:Pytho ...
- Python XML解析之DOM
DOM说明: DOM:Document Object Model API DOM是一种跨语言的XML解析机制,DOM把整个XML文件或字符串在内存中解析为树型结构方便访问. https://docs. ...
- Ubuntu17.04 sudo apt-get update升级错误
最近在折腾Ubuntu,安装的是17.04版本的.想安装PHP7.X最新版本,但是要先升级.利用sudo apt-get update命名后,出现了以下报错: 忽略:1 http://cn.archi ...
- keepalived+nginx负载均衡+ApacheWeb实现高可用
1.Keepalived高可用软件 Keepalived软件起初是专为LVS负载均衡软件设计的,用来管理并监控LVS集群系统中各个服务节点的状态,后来又加入了可以实现高可用的VRRP功能.因此,kee ...
- 批处理基础知识-IF
本文主要介绍批处理IF命令的使用. IF命令格式: if /i string=string command 释义:判断2个字符串是否相等,但不区分大小写. 例: 代码: @echo off if /i ...
- git冲突解决办法合集
一 换行符CRLF错误解决办法 1 错误产生原因 不同的操作系统使用的换行符是不一样的. unix/linux使用的是LF,max后期也采用了LF,但在windows一直采用的CRLF(回车)换行符. ...
- Git的可视化工具SourceTree管理代码 SourceTree的使用
出处:https://blog.csdn.net/android_zhengyongbo/article/details/72885860 其他参考资料https://www.cnblogs.com/ ...
- Python开发【第一篇】基础题目一
1.求1-2+3-4+5.....99的所有数的和 n = 1 s = 0 while n<100: temp = n%2 if temp == 0: #偶数 s = s-n else: s = ...