背景
工作中经常会碰到各种奇奇怪怪的需求,比如从配置文件读取出来的值转换为对象。
万能的字符串
任何时候都可以使用字符串作为属性的值,从配置文件中读取出来,如下:
配置文件内容为
test:
person: 1,张三,15
通过,分割,分别是id、姓名和年龄。
但是我们希望将它转换成一个Person对象。
@Value("${test.person}")
private Person person;
我们希望通过这种方式读取出配置文件,并转换为Person对象。
自定义转换类
PropertyEditor
这是JDK中提供的类型转化工具类
首先,定义一个实体类
package com.fandf.demo.propertyeditor;
import lombok.Data;
/**
* 在Spring源码中,有可能需要把String转成其他类型,所以在Spring源码中提供了一些技术来更方便的做对象的类型转化
*
* @author fandongfeng
*/
@Data
public class Person {
private Long id;
private String name;
private Integer age;
}
定义一个PropertyEditor
package com.fandf.demo.propertyeditor;
import java.beans.PropertyEditor;
import java.beans.PropertyEditorSupport;
/**
* 这是JDK中提供的类型转化工具类
*
* @author fandongfeng
*/
public class StringToPersonPropertyEditor extends PropertyEditorSupport implements PropertyEditor {
@Override
public void setAsText(String text) throws IllegalArgumentException {
Person person = new Person();
String[] split = text.split(",");
person.setId(Long.valueOf(split[0]));
person.setName(split[1]);
person.setAge(Integer.valueOf(split[2]));
this.setValue(person);
}
//打印结果
// Person(id=1, name=张三, age=16)
public static void main(String[] args) {
StringToPersonPropertyEditor propertyEditor = new StringToPersonPropertyEditor();
propertyEditor.setAsText("1,张三,15");
Person value = (Person) propertyEditor.getValue();
System.out.println(value);
}
}
那么在spring中应该怎么用呢? 只需要加入下面代码即可
@Bean
public CustomEditorConfigurer customEditorConfigurer() {
CustomEditorConfigurer customEditorConfigurer = new CustomEditorConfigurer();
Map<Class<?>, Class<? extends PropertyEditor>> propertyEditorMap = new HashMap<>();
// 表示StringToPersonPropertyEditor可以将String转化成Person类型,在Spring源码中,如果发现当前对象是String,而需要的类型是Person,就会使用该PropertyEditor来做类型转化
propertyEditorMap.put(Person.class, StringToPersonPropertyEditor.class);
customEditorConfigurer.setCustomEditors(propertyEditorMap);
return customEditorConfigurer;
}
ConversionService
Spring中提供的类型转化服务,它比PropertyEditor更强大
package com.fandf.demo.propertyeditor;
import org.springframework.core.convert.TypeDescriptor;
import org.springframework.core.convert.converter.ConditionalGenericConverter;
import org.springframework.core.convert.support.DefaultConversionService;
import java.util.Collections;
import java.util.Set;
/**
* @author fandongfeng
*/
public class StringToPersonConverter implements ConditionalGenericConverter {
@Override
public boolean matches(TypeDescriptor sourceType, TypeDescriptor targetType) {
return sourceType.getType().equals(String.class) && targetType.getType().equals(Person.class);
}
@Override
public Set<ConvertiblePair> getConvertibleTypes() {
return Collections.singleton(new ConvertiblePair(String.class, Person.class));
}
@Override
public Object convert(Object source, TypeDescriptor sourceType, TypeDescriptor targetType) {
Person person = new Person();
String[] split = String.valueOf(source).split(",");
person.setId(Long.valueOf(split[0]));
person.setName(split[1]);
person.setAge(Integer.valueOf(split[2]));
return person;
}
public static void main(String[] args) {
DefaultConversionService conversionService = new DefaultConversionService();
conversionService.addConverter(new StringToPersonConverter());
Person value = conversionService.convert("1,张三,15", Person.class);
System.out.println(value);
}
}
在spring中可以这样使用
@Bean
public ConversionServiceFactoryBean conversionService() {
ConversionServiceFactoryBean conversionServiceFactoryBean = new ConversionServiceFactoryBean();
conversionServiceFactoryBean.setConverters(Collections.singleton(new StringToPersonConverter()));
return conversionServiceFactoryBean;
}
TypeConverter
整合了PropertyEditor和ConversionService的功能,是Spring内部用的
SimpleTypeConverter typeConverter = new SimpleTypeConverter();
typeConverter.registerCustomEditor(Person.class, new StringToPersonPropertyEditor());
Person value = typeConverter.convertIfNecessary("1,张三,16", Person.class);
System.out.println(value);