一、代码示例

  1、我们在之前的Person类里新增一个两个属性,分别是客户的兴趣和生日,兴趣爱好有很多,我们使用list进行保存,生日使用日期进行保存

public class Person {
private String name;
public Date birth;
public Date getBirth() {
return birth;
} public void setBirth(Date birth) {
this.birth = birth;
} //兴趣爱好
public List<String> interests; public List<String> getInterests() {
return interests;
} public void setInterests(List<String> interests) {
this.interests = interests;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public void sayHello(){
System.out.println("hello "+this.name);
} }

  2、在bean里我们注入这两个参数

    <bean name="person" class="com.zjl.Person">
<property name="name" value="zhangsan"></property>
<property name="interests" value="足球,篮球"></property>
<property name="birth" value="2015-01-01"></property>
</bean>

  3、测试代码,我们打印出zhangsan的兴趣和生日

public class Test {
public static void main(String[] args) {
ClassPathXmlApplicationContext context=new ClassPathXmlApplicationContext("bean.xml");
Person person=(Person)context.getBean("person");
System.out.println(person.interests);
System.out.println(person.birth);
}
}

  4、运行结果,很不幸,我们收到了一个异常信息,提示不能将字符串转为日期格式

Cannot convert value of type [java.lang.String] to required type [java.util.Date] for property 'birth': no matching editors or conversion strategy found

  5、回看第七章源码部分,我们在源码的第10部分有如下代码,从系统获取一个conversionService,并将它放入到beanFactory中去,应该是转化,我们找到conversionService的定义方法:

        //查找是否有id为conversionService的bean,如果有,设置进beanFactory
if (beanFactory.containsBean(CONVERSION_SERVICE_BEAN_NAME) &&
beanFactory.isTypeMatch(CONVERSION_SERVICE_BEAN_NAME, ConversionService.class)) {
beanFactory.setConversionService(
beanFactory.getBean(CONVERSION_SERVICE_BEAN_NAME, ConversionService.class));
}

  6、我们试着写一个这样的bean

<bean id="conversionService" class="com.zjl.MyConversionService"></bean>

  7、类的构造如下

public class MyConversionService implements ConversionService {

    @Override
public boolean canConvert(Class<?> sourceType, Class<?> targetType) {
//判断目标类型是否是Date
if(Date.class.isAssignableFrom(targetType)){
return true;
}
System.out.println(targetType);
return false;
} @Override
public boolean canConvert(TypeDescriptor sourceType, TypeDescriptor targetType) { //判断目标类型是否是Date
if(Date.class.isAssignableFrom(targetType.getObjectType())){
return true;
}
// System.out.println(targetType);
return false;
} @Override
public <T> T convert(Object source, Class<T> targetType) {
// System.out.println("convert");
return null;
} @Override
public Object convert(Object source, TypeDescriptor sourceType, TypeDescriptor targetType) {
//如果源类型是string,我们直接将他转化为Date类型
if(String.class.isAssignableFrom(sourceType.getObjectType())){
DateFormat format=new SimpleDateFormat("yyyy-MM-dd");
try {
return format.parse((String) source);
} catch (ParseException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
// System.out.println("convert1");
return null;
} }

  8、打印结果

[足球,篮球]
Thu Jan 01 00:00:00 CST 2015

  到这里,例子基本已经完成了,可是仔细观察,我们会发现其实还有些不完美的地方:

  1、我们写入了足球,篮球,作为两个兴趣,可是程序直接将他变成了一个爱好,也就是list.add("足球,篮球"),与我们预想不一致。解决的思路我们可以想象:再注入一个bean,将字符串按照指定字符分割,转为list,

  2、由于spirng默认只能读取conversionService,我们成功转化了字符串为日期,如果想完成第一步的转化就出现了问题,我们不妨将多个conversion方法注入到bean-conversionService中,然后他依次调用和选择

二、源码分析

  1、我们看下spring中如何使用conversionService和帮我们实现一些预制的转化方法的,将我们自己定义的converter也注入进去

   <bean id="conversionService" class="org.springframework.context.support.ConversionServiceFactoryBean">
<property name="converters">
<bean class="com.zjl.MyConverter">
</bean>
</property>
</bean>

   2、我们初始化bean的时候,跟踪代码到这里,获取了系统注入的conversionService

        ConversionService conversionService = this.propertyEditorRegistry.getConversionService();
if (editor == null && conversionService != null && newValue != null && typeDescriptor != null) {
TypeDescriptor sourceTypeDesc = TypeDescriptor.forObject(newValue);
if (conversionService.canConvert(sourceTypeDesc, typeDescriptor)) {
try {
return (T) conversionService.convert(newValue, sourceTypeDesc, typeDescriptor);
}
catch (ConversionFailedException ex) {
// fallback to default conversion logic below
conversionAttemptEx = ex;
}
}
}

  3、到canConvert为在service中获取指定源格式和目标格式的converter,判断是否可以获取

    public boolean canConvert(TypeDescriptor sourceType, TypeDescriptor targetType) {
Assert.notNull(targetType, "targetType to convert to cannot be null");
if (sourceType == null) {
return true;
}
GenericConverter converter = getConverter(sourceType, targetType);
return (converter != null);
}

  4、在缓存中获取converter,如果没有,到set中获取,保存到缓存中,如果set也没有获取,保存为NO_MATCH

    protected GenericConverter getConverter(TypeDescriptor sourceType, TypeDescriptor targetType) {
ConverterCacheKey key = new ConverterCacheKey(sourceType, targetType);
GenericConverter converter = this.converterCache.get(key);
if (converter != null) {
return (converter != NO_MATCH ? converter : null);
} converter = this.converters.find(sourceType, targetType);
if (converter == null) {
converter = getDefaultConverter(sourceType, targetType);
} if (converter != null) {
this.converterCache.put(key, converter);
return converter;
} this.converterCache.put(key, NO_MATCH);
return null;
}

  5、找到converter后,调用conversionService.convert(newValue, sourceTypeDesc, typeDescriptor),如果没有converter就直接抛出错误

        GenericConverter converter = getConverter(sourceType, targetType);
if (converter != null) {
Object result = ConversionUtils.invokeConverter(converter, source, sourceType, targetType);
return handleResult(sourceType, targetType, result);
}
return handleConverterNotFound(source, sourceType, targetType);

  6、调用ConversionUtils.invokeConverter,调用converter的convert的方法

    public static Object invokeConverter(GenericConverter converter, Object source, TypeDescriptor sourceType,
TypeDescriptor targetType) {
try {
return converter.convert(source, sourceType, targetType);
}
catch (ConversionFailedException ex) {
throw ex;
}
catch (Exception ex) {
throw new ConversionFailedException(sourceType, targetType, source, ex);
}
}

  7、至于convert方法中如何进行转化就全靠我们自己写了

三、总结

  对于spring的IOC中注入的参数,虽然都是字符串,但是经过系统提供的接口我们可以将它与bean中字段的各种类型进行适配,适配过程需要定义conversionService,spring提供了默认的实现FactoryBean,他可以以set形式注入自定义的converter,也使用系统默认的转换器。

  我们来改造我们之前的转换器,通过源代码可以看到Converter以泛型中的类型作为是否对此次数据转换的选择  

public class MyConverter implements Converter<String,Date> {

    @Override
public Date convert(String source) {
DateFormat format=new SimpleDateFormat("yyyy-MM-dd");
try {
return format.parse((String) source);
} catch (ParseException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return null;
} }

  

[spring源码学习]十、IOC源码-conversionService的更多相关文章

  1. Spring Ioc源码分析系列--Ioc源码入口分析

    Spring Ioc源码分析系列--Ioc源码入口分析 本系列文章代码基于Spring Framework 5.2.x 前言 上一篇文章Spring Ioc源码分析系列--Ioc的基础知识准备介绍了I ...

  2. spring cloud深入学习(四)-----eureka源码解析、ribbon解析、声明式调用feign

    基本概念 1.Registe 一一服务注册当eureka Client向Eureka Server注册时,Eureka Client提供自身的元数据,比如IP地址.端口.运行状况指标的Uri.主页地址 ...

  3. 【 js 基础 】【 源码学习 】backbone 源码阅读(一)

    最近看完了 backbone.js 的源码,这里对于源码的细节就不再赘述了,大家可以 star 我的源码阅读项目(https://github.com/JiayiLi/source-code-stud ...

  4. 【 js 基础 】【 源码学习 】backbone 源码阅读(二)

    最近看完了 backbone.js 的源码,这里对于源码的细节就不再赘述了,大家可以 star 我的源码阅读项目(source-code-study)进行参考交流,有详细的源码注释,以及知识总结,同时 ...

  5. 【 js 基础 】【 源码学习 】backbone 源码阅读(三)浅谈 REST 和 CRUD

    最近看完了 backbone.js 的源码,这里对于源码的细节就不再赘述了,大家可以 star 我的源码阅读项目(https://github.com/JiayiLi/source-code-stud ...

  6. go 源码学习之---Tail 源码分析

    已经有两个月没有写博客了,也有好几个月没有看go相关的内容了,由于工作原因最近在做java以及大数据相关的内容,导致最近工作较忙,博客停止了更新,正好想捡起之前go的东西,所以找了一个源码学习 这个也 ...

  7. 【 js 基础 】【 源码学习 】backbone 源码阅读(三)

    最近看完了 backbone.js 的源码,这里对于源码的细节就不再赘述了,大家可以 star 我的源码阅读项目(https://github.com/JiayiLi/source-code-stud ...

  8. Jetty源码学习-编译Jetty源码二三事

    工作小几个月了,JDK基础和web应用框架学的的差不多了,开始学习Jetty源码,费了小半天才编译成功,把自己拆过的坑记录下来. 编译前的环境: MAVEN 3.3.Eclips eLuna Serv ...

  9. Spring源码学习之IOC容器实现原理(一)-DefaultListableBeanFactory

    从这个继承体系结构图来看,我们可以发现DefaultListableBeanFactory是第一个非抽象类,非接口类.实际IOC容器.所以这篇博客以DefaultListableBeanFactory ...

随机推荐

  1. JavaScript面向对象和原型函数

    对象,是javascript中非常重要的一个梗,是否能透彻的理解它直接关系到你对整个javascript体系的基础理解,说白了,javascript就是一群对象在搅..(哔!). 常用的几种对象创建模 ...

  2. WCF技术内幕 第二章 - 简单的Message

    1.契约 - 接口 (客户端和服务端都要认识Message) namespace WCFService { [ServiceContract(Namespace = "http://wint ...

  3. logging 文件日志

    1. 例子 import logging logging.basicConfig(filename='log.txt', #文件名 level=logging.DEBUG, #级别 format=u' ...

  4. 转载:Centos7 从零编译Nginx+PHP+MySql 序言 一

    这次玩次狠得.除了编译器使用yum安装,其他全部手动编译.哼~ 看似就Nginx.PHP.MySql三个东东,但是它们太尼玛依赖别人了. 没办法,想用它们就得老老实实给它们提供想要的东西. 首先的一些 ...

  5. Jquery客户端校验——jquery.validate.js

    jQuery Validate 插件为表单提供了强大的验证功能,让客户端表单验证变得更简单,同时提供了大量的定制选项,满足应用程序各种需求.该插件捆绑了一套有用的验证方法,包括 URL 和电子邮件验证 ...

  6. MySQL主从复制

    Mysql主从复制介绍 MySQL支持单向.双向.链式级联.实时.异步复制.在复制过程中,一台服务器充当服务器(Master),而一个或多个其它的服务器充当从服务器(Slave). 复制可以是单向:M ...

  7. SubMenu的setHeaderView使用时发现的问题

    SubMenu android.view.SubMenu.setHeaderView(View view) 上面是这个方法的完整签名,作用就是自定义子菜单的菜单头,但是在OptionsMenu里面设置 ...

  8. bootstarp

    我最近在学一个前端框架叫bootstarp.我的老大给了一个bootstarp的模板给我,我看了看感觉没意思.因为那里面的样式,我也会写,所以我就没心思去看,每次只要面板上有的我就搬下来就完了,只要面 ...

  9. Android中矢量动画

    Android中矢量动画 Android中用<path> 标签来创建SVG,就好比控制着一支画笔,从一点到一点,动一条线. <path> 标签 支持一下属性 M = (Mx, ...

  10. libvirt 网络手册(一)

    如果选择网络类型 在一个专用的服务器上,虚拟机常常需要被从公网访问(也就是每个虚拟机都需要公网地址),这时就需要桥接网络,它使得每个虚拟机有自己的IPV4和IPV6地址.如果桥接不可用,可以创建一个R ...