SpringMVC如何实现数据类型转换
1、类型转换:首先表单数据(全部是字符串)通过WebDataBinder进行绑定到命令对象,内部通过Converter SPI实现;
2、数据验证:使用JSR-303验证框架进行验证;
3、格式化显示:在表单页面可以通过如下方式展示通过内部通过Converter SPI格式化的数据和错误信息;- 数据类型转换
SpringMVC数据转换包括两部分
一、类型转换器:提供类型转换的实现支持;
1、Converter:类型转换器,用于转换S 类型到T 类型,此接口的实现必须是线程安全的且可以被共享。
package org.springframework.core.convert.converter;
public interface Converter<S, T> {
// S 是源类型T 是目标类型T convert(S source);
// 转换S 类型的source 到T 目标类型的转换方法
}
2、GenericConverter 和ConditionalGenericConverter:GenericConverter 接口实现能在多种类型之间进行转换,ConditionalGenericConverter 是有条件的在多种类型之间进行转换。GenericConverter:
package org.springframework.core.convert.converter;
public interface GenericConverter {
Set<ConvertiblePair> getConvertibleTypes();
Object convert(Object source, TypeDescriptor sourceType, TypeDescriptor targetType);
}
- getConvertibleTypes:指定了可以转换的目标类型对;convert:在sourceType 和targetType 类型之间进行转换。
ConditionalGenericConverter:
package org.springframework.core.convert.converter;
public interface ConditionalGenericConverter extends GenericConverter {
boolean matches(TypeDescriptor sourceType, TypeDescriptor targetType);
}
- matches:用于判断sourceType 和targetType 类型之间能否进行类型转换。
3、ConverterFactory:工厂模式的实现,用于选择将一种S 源类型转换为R 类型的子类型T 的转换器的工厂接口。
package org.springframework.core.convert.converter;
public interface ConverterFactory<S, R> {
<T extends R> Converter<S, T> getConverter(Class<T> targetType);
//S:源类型;R 目标类型的父类型;T:目标类型,且是R 类型的子类型;
}
二、类型转换器注册器、类型转换服务:提供类型转换器注册支持,运行时类型转换API 支持,运行时类型转换API 支持。
1、ConverterRegistry:类型转换器注册支持,可以注册/删除相应的类型转换器。
可以注册:Converter 实现,GenericConverter 实现,ConverterFactory 实现。
2、ConversionService:运行时类型转换服务接口,提供运行期类型转换的支持。
Spring 提供了两个默认实现(其都实现了ConverterRegistry、ConversionService 接口):DefaultConversionService:默认的类型转换服务实现;DefaultFormattingConversionService:带数据格式化支持的类型转换服务实现,一般使用该服务实现即可。
Spring内建的数据类型转换器
一、标量转换器
StringToBooleanConverter | String----->Boolean true:true/on/yes/1;false:false/off/no/0 |
ObjectToStringConverter | Object----->String调用toString方法转换 |
StringToNumberConverterFactory | String----->Number(如Integer、Long 等) |
NumberToNumberConverterFactory | Number 子类型(Integer、Long、Double 等)<——> Number 子类型(Integer、Long、Double 等) |
StringToCharacterConverter | String----->java.lang.Character取字符串第一个字符 |
NumberToCharacterConverter | java.lang.Character——>Number 子类型(Integer、Long、Double 等) |
StringToEnumConverterFactory | String----->enum类型通过Enum.valueOf将字符串转换为需要的enum类型 |
EnumToStringConverter | enum类型----->String返回enum对象的name() |
StringToLocaleConverter | String----->java.util.Local |
PropertiesToStringConverter | java.util.Properties----->String默认通过ISO-8859-1 解码 |
StringToPropertiesConverter | String----->java.util.Properties默认使用ISO-8859-1 编码 |
二、集合、数组相关转换器
类名 | 说明 |
ArrayToCollectionConverter | 任意S 数组---->任意T 集合(List、Set) |
CollectionToArrayConverter | 任意T 集合(List、Set)---->任意S 数组 |
ArrayToArrayConverter | 任意S 数组<---->任意T 数组 |
CollectionToCollectionConverter | 任意T 集合(List、Set)<---->任意T 集合(List、Set)即集合之间的类型转换 |
MapToMapConverter | Map<---->Map 之间的转换 |
ArrayToStringConverter | 任意S 数组---->String 类型 |
StringToArrayConverter | String----->数组默认通过“,” 分割,且去除字符串的两边空格(trim) |
ArrayToObjectConverter | 任意S 数组---->任意Object 的转换(如果目标类型和源类型兼容,直接返回源对象;否则返回S 数组的第一个元素并进行类型转换) |
ObjectToArrayConverter | Object----->单元素数组 |
CollectionToStringConverter | 任意T 集合(List、Set)---->String 类型 |
StringToCollectionConverter | String----->集合(List、Set)默认通过“,” 分割,且去除字符串的两边空格(trim) |
CollectionToObjectConverter | 任意T 集合---->任意Object 的转换(如果目标类型和源类型兼容,直接返回源对象;否则返回S 数组的第一个元素并进行类型转换) |
ObjectToCollectionConverter | Object----->单元素集合 |
自定义数据类型转换器
一、StringToDateConverter:自定义String类型转Date类型的转换类;
二、StringToPhoneNumberConverter:自定义String类型转PhoneNumberModel类型的转换类
1、注册ConversionService 实现和自定义的类型转换器
2、通过ConfigurableWebBindingInitializer 注册ConversionService
3、注册ConfigurableWebBindingInitializer 到RequestMappingHandlerAdapter
如果是使用的<mvc:annotation-drive>的方式,那么上面的第2和3步就不用做了,直接在<mvc:annotation-drive>里面配置conversion-service属性即可,示例:
<mvc:annotation-driven conversion-service=“conversionService“ />
配置自定义的数据类型转换器
-