1. 揭秘Spring类型转换 - 框架设计的基石
仰不愧天,俯不愧人,内不愧心。关注公众号【BAT的乌托邦】,有Spring技术栈、MyBatis、JVM、中间件等小而美的原创专栏供以免费学习。分享、成长,拒绝浅尝辄止。本文已被 https://www.yourbatman.cn 收录。
✍前言
你好,我是YourBatman。
Spring Framework
是一个现代化的框架,俨然已发展成为Java开发的基石。随着高度封装、高度智能化的Spring Boot的普及,发现团队内越来越少的人知道其深层次机制,哪怕只有一点点。这是让Spirng团队开心,但却是让使用的团队比较担忧的现象。
若运行一个完全黑箱程序无疑像抱着一个定时炸弹,总是如履薄冰、战战兢兢。团队内需要这样的同学来为它保驾护航,惊爆之时方可泰然自诺。所以,你愿意pick吗?
本系列将讨论Spring Framework
里贯穿其上下文,具有举足轻重地位的一个模块:类型转换(也可叫数据转换)。
✍正文
Java是个多类型且强类型语言,类型转换这个概念对它来说并不陌生。比如:
- 自动类型转换(隐式):小类型 -> 大类型。eg:
int a = 10; double b = a;
- 强制类型转换(显式):大类型 -> 小类型。eg:
double a = 10.123; int b = (int)a;
- 说明:强转有可能产生精度丢失
- 调用API类型转换:常见的是字符串和其它类型的互转。eg:
parseInt(String); parseBoolean(String); JSON.toJSONString(Obj); LocalDate.parse(String)
- 说明:API可能来自于JDK提供、一方库、二方库、三方库提供
在企业级开发环境中,会遇到更为复杂的数据转换场景,譬如说:
- 输入/传入一个规格字符串(如
1,2,3,4
),转换为一个数组 - 输入/传入一个JSON串(如
{"name":"YourBatman","age":18}
),转换为一个Person对象 - 输入/传入一个URL串(如:
C:/myfile.txt、classpath:myfile.txt
),转换为一个org.springframework.core.io.Resource
对象
虽说数据输入/传入绝大部分都会是字符串(如Http请求信息、XML配置信息),但结构可以千差万别,那么这就必然会涉及到大量的数据类型、结构转换的逻辑。倘若这都需要程序员自己手动编码做转换处理,那会让人望而生畏甚至怯步。
还好我们有Spring。从本文起,A哥就帮你解密Spring Framework它是如何帮你接管类型转换,实现“自动化”的。有了此部分知识的储备,后续再讨论自动化数据绑定、自动化数据校验、Spring Boot松散绑定等,一切都变得容易接受得多。
说明:类型转换其实每个框架都会存在,其中Java领域以Spring的实现最为经典,学会后便可举一反三
Spring类型转换
Spring的类型转换也并非一步到位。完全掌握Spring的类型转换并非易事,需要有一定的脉络按步骤进行。本文作为类型转换系列第一篇文章,将绘制目录大纲,将从以下几个方面逐步展开讨论。
早期类型转换之PropertyEditor
早期的Spirng(3.0之前)类型转换是基于Java Beans接口java.beans.PropertyEditor
来实现的(全部继承自PropertyEditorSupport
):
public interface PropertyEditor {
...
// String -> Object
void setAsText(String text) throws java.lang.IllegalArgumentException;
// Object -> String
String getAsText();
...
}
这类实现举例有:
StringArrayPropertyEditor
:,
分隔的字符串和String[]
类型互转PropertiesEditor
:键值对字符串和Properties
类型互转IntegerEditor
:字符串和Integer
类型互转- ...
基于PropertyEditor
的类型转换作为一种古老的、遗留下来的方式,是具有一些设计缺陷的,如:职责不单一,类型不安全,只能实现String
类型的转换等。虽然自Spring 3.0起提供了现代化的类型转换接口,但是此部分机制一直得以保留,保证了向下兼容性。
说明:Spring 3.0之前在Java领域还未完全站稳脚跟,因此良好的向下兼容显得尤为重要
这块内容将在本系列后面具体篇章中得到专题详解,敬请关注。
新一代类型转换接口Converter、GenericConverter
为了解决PropertyEditor
作为类型转换方式的设计缺陷,Spring 3.0版本重新设计了一套类型转换接口,其中主要包括:
Converter<S, T>
:Source -> Target类型转换接口,适用于1:1转换- StringToPropertiesConverter:将String类型转换为Properties
- StringToBooleanConverter:将String类型转换为Boolean
- EnumToIntegerConverter:将Enum类型转换为Integer
ConverterFactory<S, R>
:Source -> R类型转换接口,适用于1:N转换- StringToEnumConverterFactory:将String类型转任意Enum
- StringToNumberConverterFactory:将String类型转为任意数字(可以是int、long、double等等)
- NumberToNumberConverterFactory:数字类型转为数字类型(如int到long,long到double等等)
GenericConverter
:更为通用的类型转换接口,适用于N:N转换- ObjectToCollectionConverter:任意集合类型转为任意集合类型(如
List<String>
转为List<Integer> / Set<Integer>
都使用此转换器) - CollectionToArrayConverter:解释基本同上
- MapToMapConverter:解释基本同上
- ObjectToCollectionConverter:任意集合类型转为任意集合类型(如
ConditionalConverter
:条件转换接口。可跟上面3个接口组合使用,提供前置条件判断验证
重新设计的这套接口,解决了PropertyEditor
做类型转换存在的所有缺陷,且具有非常高的灵活性和可扩展性。但是,每个接口独立来看均具有一定的局限性,只有使用组合拳方才有最大威力。当然喽,这也造成学习曲线变得陡峭。据我了解,很少有同学搞得清楚新的这套类型转换机制,特别容易混淆。倘若你掌握了是不是自己价值又提升了呢?不信你细品?
这块内容将在本系列后面具体篇章中得到专题详解,敬请关注。
新一代转换服务接口:ConversionService
从上一小节我们知道,新的这套接口中,Converter、ConverterFactory、GenericConverter
它们三都着力于完成类型转换。对于使用者而言,如果做个类型转换需要了解到这三套体系无疑成本太高,因此就有了ConversionService
用于整合它们三,统一化接口操作。
此接口也是Spring 3.0新增,用于统一化 底层类型转换实现的差异,对外提供统一服务,所以它也被称作类型转换的门面接口,从接口名称xxxService
也能看出来其设计思路。它主要有两大实现:
GenericConversionService
:提供模版实现,如转换器的注册、删除、匹配查找等,但并不内置转换器实现DefaultConversionService
:继承自GenericConversionService。在它基础上默认注册了非常多的内建的转换器实现,从而能够实现绝大部分的类型转换需求
ConversionService
转换服务它贯穿于Spring上下文ApplicationContext
的多项功能,包括但不限于:BeanWrapper处理Bean属性、DataBinder数据绑定、PropertySource外部化属性处理等等。因此想要进一步深入了解的话,ConversionService是你绕不过去的坎。
说明:很多小伙伴问WebConversionService是什么场景下使用?我说:它并非Spirng Framework的API,而属于Spring Boot提供的增强,且起始于2.x版本,这点需引起注意
这块内容将在本系列后面具体篇章中得到专题详解,敬请关注。
类型转换整合格式化器Formatter
Spring 3.0还新增了一个Formatter<T>
接口,作用为:将Object格式化为类型T。从语义上理解它也具有类型转换(数据转换的作用),相较于Converter<S,T>
它强调的是格式化,因此一般用于时间/日期、数字(小数、分数、科学计数法等等)、货币等场景,举例它的实现:
DurationFormatter
:字符串和Duration
类型的互转CurrencyUnitFormatter
:字符串和javax.money.CurrencyUnit
货币类型互转DateFormatter
:字符串和java.util.Date
类型互转。这个就使用得太多了,它默认支持什么格式?支持哪些输出方式,这将在后文详细描述- ......
为了和类型转换服务ConversionService
完成整合,对外只提供统一的API。Spring提供了FormattingConversionService
专门用于整合Converter和Formatter,从而使得两者具有一致的编程体验,对开发者更加友好。
这块内容将在本系列后面具体篇章中得到专题详解,敬请关注。
类型转换底层接口TypeConvert
定义类型转换方法的接口,它在Spring 2.0就已经存在。在还没有ConversionService
之前,它的类型转换动作均委托给已注册的PropertyEditor
来完成。但自3.0之后,这个转换动作可能被PropertyEditor来做,也可能交给ConversionService
处理。
它一共提供三个重载方法:
// @since 2.0
public interface TypeConverter {
// value:待转换的source源数据
// requiredType:目标类型targetType
// methodParam:转换的目标方法参数,主要为了分析泛型类型,可能为null
// field:目标的反射字段,为了泛型,可能为null
<T> T convertIfNecessary(Object value, Class<T> requiredType) throws TypeMismatchException;
<T> T convertIfNecessary(Object value, Class<T> requiredType, MethodParameter methodParam) throws TypeMismatchException;
<T> T convertIfNecessary(Object value, Class<T> requiredType, Field field) throws TypeMismatchException;
}
它是Spring内部使用类型转换的入口,最终委托给PropertyEditor
或者注册到ConversionService
里的转换器去完成。它的主要实现有:
TypeConverterSupport
:@since 3.2。继承自PropertyEditorRegistrySupport
,它主要是为子类BeanWrapperImpl
提供功能支撑。作用有如下两方面:- 提供对默认编辑器(支持JDK内置类型的转换如:Charset、Class、Class[]、Properties、Collection等等)和自定义编辑器的管理(PropertyEditorRegistry#registerCustomEditor)
- 提供get/set方法,把
ConversionService
管理上(可选依赖,可为null)
- 数据绑定相关:因为数据绑定强依赖于类型转换,因此数据绑定涉及到的属性访问操作将会依赖于此组件,不管是直接访问属性的
DirectFieldAccessor
还是功能更强大的BeanWrapperImpl
均是如此
总的来说,TypeConverter
能把类型的各种实现、API收口于此,Spring把类型转换的能力都转嫁到TypeConverter这个API里面去了。虽然方便了使用,但其内部实现原理稍显复杂,同样的这块内容将在本系列后面具体篇章中得到专题详解,敬请关注。
Spring Boot使用增强
在传统Spring Framework场景下,若想使用ConversionService
还得手动档去配置,这对于不太了解其运行机制的同学无疑是有使用门槛的。而在Spring Boot场景下这一切都会变得简单许多,可谓使用起来愈发方便了。
另外,Spring Boot在内建转换器的基础上额外扩展了不少实用转换器,形如:
StringToFileConverter
:String -> FileNumberToDurationConverter
:DelimitedStringToCollectionConverter
:- ......
✍总结
基于配置来控制程序运行总比你修改程序代码来得更优雅、更富弹性,但这是需要依赖于数据绑定、数据校验等功能的,而它们又依赖于类型转换。
虽说几乎所有的框架都会有类型转换的功能模块,但Spring的可能是最为通用、最为经典的存在。因此本系列专题讲解Spring Framework的类型转换,旨在能够帮你你撬开通往跃升的大门,节节攀高。
推荐阅读:
- Spring Boot 2.4.0正式发布,全新的配置文件加载机制(不向下兼容)
- 如果程序员和产品经理都用凡尔赛文学对话......
- Spring Framework 5.3.0正式发布,在云原生路上继续发力
- Spring改变版本号命名规则:此举对非英语国家很友好
- JDK15正式发布,划时代的ZGC同时宣布转正
- IntelliJ IDEA 2020.2正式发布,诸多亮点总有几款能助你提效
- Spring Boot 2.3.0正式发布:优雅停机、配置文件位置通配符新特性一览
- 搞事情?Spring Boot今天一口气发布三个版本
1. 揭秘Spring类型转换 - 框架设计的基石的更多相关文章
- spring boot 框架设计步骤
spring boot 框架设计步骤: 1.poem.xml配置 2.application.yml配置 3.entiry实体 4.realm.Myrealm extends AuthorizingR ...
- spring MVC框架入门(外加SSM整合)
spring MVC框架 一.什么是sping MVC Spring MVC属于SpringFrameWork的后续产品,已经融合在Spring Web Flow里面.Spring 框架提供了构建 W ...
- Spring Framework框架解析(1)- 从图书馆示例来看xml文件的加载过程
引言 这个系列是我阅读Spring源码后的一个总结,会从Spring Framework框架的整体结构进行分析,不会先入为主的讲解IOC或者AOP的原理,如果读者有使用Spring的经验再好不过.鉴于 ...
- Spring Type Conversion(Spring类型转换源码探究)
1:概述 类型转换系统负责Spring框架中对象类型转换和格式化工作. ConversionService默认实现UML图如下所示: GenericConversionService(通用类型转换服务 ...
- Spring Type Conversion(Spring类型转换)
Spring Type Conversion(Spring类型转换) 1:概述: Spring3引入了core.convert包,提供了通用类型转换系统,定义了实现类型转换和运行时执行类型的SPI. ...
- 组件化框架设计之AOP&IOC(四)
阿里P7移动互联网架构师进阶视频(每日更新中)免费学习请点击:https://space.bilibili.com/474380680 本篇文章将从以下两个方面来介绍组件化框架设计: [AOP(面向切 ...
- 从零开始学 Java - 搭建 Spring MVC 框架
没有什么比一个时代的没落更令人伤感的了 整个社会和人都在追求创新.进步.成长,没有人愿意停步不前,一个个老事物慢慢从我们生活中消失掉真的令人那么伤感么?或者说被取代?我想有些是的,但有些东西其实并不是 ...
- Web信息架构——设计大型网站(第3版)(久负盛名经典再现,信息架构设计领域基石之作!)
Web信息架构——设计大型网站(第3版)(久负盛名经典再现,信息架构设计领域基石之作!) [美]]Peter Morville(彼得·莫维尔) Louis Rosenfeld(路易斯·罗森菲尔德) ...
- 【niubi-job——一个分布式的任务调度框架】----框架设计原理以及实现
引言 niubi-job的框架设计是非常简单实用的一套设计,去掉了很多其它调度框架中,锦上添花但并非必须的组件,例如MQ消息通讯组件(kafka等).它的框架设计核心思想是,让每一个jar包可以相对之 ...
随机推荐
- Java安全框架(一)Spring Security
Java安全框架(一)Spring Security 文章主要分三部分 1.Spring Security的架构及核心组件:(1)认证:(2)权限拦截:(3)数据库管理:(4)权限缓存:(5)自定 ...
- 【漏洞复现】Shiro<=1.2.4反序列化漏洞
0x01 概述 Shiro简介 Apache Shiro是一个强大且易用的Java安全框架,执行身份验证.授权.密码和会话管理.使用Shiro的易于理解的API,您可以快速.轻松地获得任何应用程序,从 ...
- OpenCV计算机视觉学习(10)——图像变换(傅里叶变换,高通滤波,低通滤波)
如果需要处理的原图及代码,请移步小编的GitHub地址 传送门:请点击我 如果点击有误:https://github.com/LeBron-Jian/ComputerVisionPractice 在数 ...
- Pycharm激活码无偿分享,2020年最新Pycharm永久激活码!
2020年10月7日08:04:34更新的Pycharm激活码,还热乎着呢,速用~ 如果下边的这个Pycharm激活码过期失效了的话,大家可以关注我的微信公众号:Python联盟,然后回复" ...
- 剑指Offer-Python(16-20)
16.合并另个排序链表 # -*- coding:utf-8 -*- class ListNode: def __init__(self, x): self.val = x self.next = N ...
- ubutun 服务器配置jupyter notebook
由于能力有限,学习机器学习时候发现,自己的电脑带不起来,所以想起了服务器,选择的是阿里的ubutun服务器,所以希望能够 使用jupyter notebook,看到网上一大片,配置和好久,才成功,在这 ...
- ngx instance
首先看下 连接池的获取以及释放 ngx_connection_t * ngx_get_connection(ngx_socket_t s, ngx_log_t *log) //从连接池中获取一个ngx ...
- java多线程---张孝祥
1.java web 中,一次http请求是一个任务,因为服务器里面有线程池的,存在一个线程处理多个请求任务. 2.在java中,vector,hashtable,concurrentHashMap是 ...
- Flink处理函数实战之四:窗口处理
欢迎访问我的GitHub https://github.com/zq2599/blog_demos 内容:所有原创文章分类汇总及配套源码,涉及Java.Docker.Kubernetes.DevOPS ...
- C++深拷贝与浅拷贝区别
浅拷贝只是对指针的拷贝,浅拷贝后两个指针指向同一个内存空间: 深拷贝不仅对指针进行拷贝,而且对指针指向的内容进行拷贝,经深拷贝后的指针是指向两个不同地址的指针. 当对一个已知对象进行拷贝时,编译系统会 ...