Spring核心——数据的类型转换

字符串到实体转换一文中介绍了Spring核心框架中使用PropertyEditor将任何字符串转换为数字、实体的方法。除了字符串到实体,Spring还提供了更加通用的功能在对象和对象之间进行数据转换。

Converter<S, T>

Spring的类型转换的基础是Converter<S, T>(以下简称转换器)接口:

packageorg.springframework.core.convert.converter;publicinterfaceConverter{Tconvert(S source);}

光是看他的结构就很清晰的明白这个接口是要做什么。S表示Source(来源)、T表示Target(目标),所以这个接口的2个范型参数就是数据从S转换为T,Converter::convert方法正是输入一个“S”类型的实例,返回一个“T”类型的实例。

可以通过这个接口实现规范化、可复用的类型转换功能。下面通过转换器实现字符串到PC实体类相互转换的过程。

Pc实体:

publicclassPCextendsDevice{String cpu;String graphic;String ram;//Getter & Setter ...}

在基类Device中通过反射实现字符串到实体类的转换:

publicabstractclassDevice{publicvoidpares(String text){//字符串转换为实体Field[] fields =this.getClass().getDeclaredFields();for(Field field : fields) {intbegIndex = text.indexOf(field.getName());intendIndex = text.indexOf(";", begIndex);String sub = text.substring(begIndex, endIndex), value = sub.split("=")[1];field.setAccessible(true);    field.set(this, value);}};publicStringvalue(){//实体转换为字符串Field[] fields =this.getClass().getDeclaredFields();StringBuilder sb =newStringBuilder();for(Field field : fields) {sb.append(field.getName());sb.append("=");sb.append(field.get(this).toString());sb.append(";");}returnsb.toString();}}

然后声明两个转换器的实现类:

publicclassString2PcConverterimplementsConverter{//字符串转换为PC对象@OverridepublicPCconvert(String source){PC pc =newPC();pc.pares(source);returnpc;}}

publicclassPC2StringConverterimplementsConverter{//PC对象转换为字符串@OverridepublicStringconvert(PC source){returnsource.value();}}

 最后使用这两个转换器:

publicclassConversionApp{voidsingletonConversion(){finalString text ="cpu=amd;ram=kingston;graphic=Navidia;";Converter string2Pc =newString2PcConverter();PC pc = string2Pc.convert(text);Converter pc2String =newPC2StringConverter();String string = pc2String.convert(pc);}}

以上就是Spring最基本的类型转换功能——围绕着转换器(Converter<S, T>)接口实现数据类型转换。看到这里可能有些码友就要问了:这到底有什么用?直接用使用Device::pares和Device::value方法不就完事了?为什么还要引入转换器兜一圈??!

如果系统仅仅只有1个或几个类型转换确实没必要引入转换器。但是业务总是繁杂多样的,模块与模块之前也会存在数据结构的差异,因此我们需要适配器(Adapter)、外观(Facade)等模式来应对变化多端的外部输入而无需改动业务逻辑。实际上从更高的层次看,Converter接口就是Spring为类型转换提供的一个适配器。后面会看到Spring已经为程序的顺利运行提供了大量的转换器,即使在阅读本文内容之前不知道这些转换器的存在,但Spring框架时时刻刻都在使用他们。

ConverterFactory<S, R>

转换器只能对单一类型进行转换,如果有大量相同类别的数据需要转换可以使用ConverterFactory(一下简称转换工厂):

publicinterfaceConverterFactory{    ConvertergetConverter(Class<T> targetType);}

ConverterFactory::getConverter是返回一个转换器,这里范型标记“T”是“R”的子类。看下面转换工厂的例子,他可以将字符串转换成Device的子类:

publicclassString2DeviceConverterFactoryimplementsConverterFactory{publicConvertergetConverter(Class<T> targetType){returnnewString2DeviceConverter(targetType);}// Device的通用转换器staticclassString2DeviceConverterimplementsConverter{privateClass klass;publicString2DeviceConverter(Class<? extends Device> klass){this.klass = klass;}publicTconvert(String source){Device device =null;device = klass.newInstance();device.pares(source);return(T) device;}}}

然后可以使用这个转换工厂按照目标类型进行转换:

publicclassConversionApp{voidfactoryConversion(){String2DeviceConverterFactory factory =newString2DeviceConverterFactory();Converter pcConverter = factory.getConverter(PC.class);//将字符串转换为PCPC pc = pcConverter.convert("cpu=amd;ram=kingston;graphic=Navidia;");Converter phoneConverter = factory.getConverter(Phone.class);//将字符串转换为PhonePhone phone = phoneConverter.convert("name=HUAWEIP20;cpu=Kirin970;ram=64G;");}}

Phone是另外一个继承了Device的实体类:

publicclassPhoneextendsDevice{String name;String cpu;String ram;// Getter & Setter}

数据转换服务

Spring已经为数据转换预设了大量的Converter,这些Converter可以通过ConversionService直接使用。ConversionService中包含了几乎所有Java常规类型的数据格式转换,看下面的案例。

publicclassConversionApp{ConversionAppregistConversionService(){ConfigurableApplicationContext ctx =newAnnotationConfigApplicationContext(ConversionConfig.class);// 获取ConversionServiceConversionService service = ctx.getBean(ConversionService.class);// 字符串转换为整型inti = service.convert("123456", Integer.class);// 字符串转换为浮点floatf = service.convert("1234.56", Float.class);// 源生列表转换为ListList list = service.convert(newint[] {1,2,3,4,5,6}, List.class);// 源生列表转换为SetSet set = service.convert(newint[] {1,2,3,4,5,6}, Set.class);// 枚举转换Gender gender = service.convert("Male", Gender.class);// 使用自定义转换器PC pc = service.convert("cpu=amd;ram=kingston;graphic=Navidia;", PC.class);// UUID转换UUID uuid = service.convert("f51b4b95-0925-4ad0-8c62-4daf3ea7918f", UUID.class);// 字符串转换为Optional<PC>Optional options = service.convert("cpu=amd;ram=kingston;graphic=Navidia;", Optional.class);// 使用TypeDescriptor描述进行转换String source ="123456789";intresult = (int) service.convert(source, TypeDescriptor.valueOf(source.getClass()),TypeDescriptor.valueOf(Integer.class));_G.print(result);}enumGender {Male, Female, Other}}

除了上面的转换,ConversionService还提供了其他转换器,详情请看org.springframework.core.convert.support.DefaultConversionService的JavaDoc文档。

需要通过ConversionServiceFactoryBean来启用ConversionService,下面的代码是在@Configurable中向IoC容器添加ConversionServiceFactoryBean:

@ConfigurablepublicclassConversionConfig{@BeanpublicConversionServiceFactoryBeanConversionServiceFactoryBean(){ConversionServiceFactoryBean factoryBean =newConversionServiceFactoryBean();Set converters =newHashSet<>();// 添加自定义转换器converters.add(newString2PcConverter());converters.add(newPC2StringConverter());factoryBean.setConverters(converters);returnfactoryBean;}}

也可以通过XML文件配置来引入ConversionService:

ConversionService在Spring MVC中的作用很大,可以全局注册统一的类型转换器。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,928评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,192评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,468评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,186评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,295评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,374评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,403评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,186评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,610评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,906评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,075评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,755评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,393评论 3 320
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,079评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,313评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,934评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,963评论 2 351

推荐阅读更多精彩内容