MyBatis接口(Bean)与配置信息(Mapper)绑定
2015-10-31 00:00
495 查看
摘要: 扫描方式提供接口(Bean)与配置信息(Mapper)绑定
目的
MyBatis的XML配置文件解析成JAVA类并在内存中存储,但是在程序运行时需要对应的类去调用,而相应的调用类还没有实例化,现在流行的都是使用Spring去管理需要的对象,Spring提供2种方式,分别为XML与注解。下面来分析调用类的实例化及与配置绑定。
1 XML方式
这里相等于实例一个id为menuMapper的对象,这个对象实际类是MapperFactoryBean,里面包含2个属性。当需要使用此对象时,直接使用id注入即可。下面来看看类里面做了哪些操作?
自身校验接口类不能为空,父类校验SqlSession不能为空,说明SqlSessionFactory与SqlSessionTemplate必须存在一个。
在MyBatis整合Spring的实现(17)中分析已经知道,SQL对应的Mapper文件已经解析并添加到Configuration(全局配置类)中,为什么这里还会添加Mapper呢?因为前面分析的是对XML的解析,如果我们的配置使用的注解方式,那么这里,我的Mapper是没有解析的,当没有解析时,这里就会去解析注解并添加到Configuration(全局配置类)中。
2 注解方式
这里使用注解方式,把包cn.vansky.schedule.time下面的所有JAVA类全部实例化,但是这样实例的话,那么实际类型就是对应JAVA类的类型,也就相当于下面的配置。
这里如果要是直接使用id注入,那么就调用不到MyBatis的配置,所以分析MapperScannerConfigurer类做了哪些操作?
MapperScannerConfigurer类实现BeanDefinitionRegistryPostProcessor,需要执行postProcessBeanDefinitionRegistry方法。上图红框中,Spring把接口MenuMapper类的属性修改了,使此类的实例化最终的Bean变成MapperFactoryBean并注入MapperFactoryBean类的属性mapperInterface。也就相当于XML的配置(查看本章节1)。
3 BEAN注入
JAVA代码里面注入使用注解@Autowired,此注解默认使用类型获取对应的对象。
这里类型MenuMapper,而实际类型为MapperFactoryBean,应该会出现类型转换异常呀!
MyBatis做了进一步处理,MapperFactoryBean实现FactoryBean,必须实现getObject方法。
这里是获取最终的对象,mapperInterface属性就是实际的类型MenuMapper。
SqlSession是SqlSessionTemplate,SqlSessionTemplate的getMapper方法。
通过代理工厂,实际的对象就通过代理实例化完成。最终的代理类就是MapperProxy。
总结
注解方式就是对XML的整合,把多个Bean实例化整合成一个XML配置,这样省去多余的配置信息,而且还提供了注解方式来配置SQL的Mapper信息。
目的
MyBatis的XML配置文件解析成JAVA类并在内存中存储,但是在程序运行时需要对应的类去调用,而相应的调用类还没有实例化,现在流行的都是使用Spring去管理需要的对象,Spring提供2种方式,分别为XML与注解。下面来分析调用类的实例化及与配置绑定。
1 XML方式
<bean id="menuMapper" class="org.mybatis.spring.mapper.MapperFactoryBean"> <property name="sqlSessionFactory" ref="sqlSessionFactory" /> <property name="mapperInterface" value="cn.vansky.schedule.time.menu.dao.MenuMapper" /> </bean>
这里相等于实例一个id为menuMapper的对象,这个对象实际类是MapperFactoryBean,里面包含2个属性。当需要使用此对象时,直接使用id注入即可。下面来看看类里面做了哪些操作?
protected void checkDaoConfig() { super.checkDaoConfig(); notNull(this.mapperInterface, "Property 'mapperInterface' is required"); Configuration configuration = getSqlSession().getConfiguration(); if (this.addToConfig && !configuration.hasMapper(this.mapperInterface)) { try { configuration.addMapper(this.mapperInterface); } catch (Throwable t) { logger.error("Error while adding the mapper '" + this.mapperInterface + "' to configuration.", t); throw new IllegalArgumentException(t); } finally { ErrorContext.instance().reset(); } } } protected void checkDaoConfig() { notNull(this.sqlSession, "Property 'sqlSessionFactory' or 'sqlSessionTemplate' are required"); } @Autowired(required = false) public final void setSqlSessionFactory(SqlSessionFactory sqlSessionFactory) { if (!this.externalSqlSession) { this.sqlSession = new SqlSessionTemplate(sqlSessionFactory); } } @Autowired(required = false) public final void setSqlSessionTemplate(SqlSessionTemplate sqlSessionTemplate) { this.sqlSession = sqlSessionTemplate; this.externalSqlSession = true; }
自身校验接口类不能为空,父类校验SqlSession不能为空,说明SqlSessionFactory与SqlSessionTemplate必须存在一个。
在MyBatis整合Spring的实现(17)中分析已经知道,SQL对应的Mapper文件已经解析并添加到Configuration(全局配置类)中,为什么这里还会添加Mapper呢?因为前面分析的是对XML的解析,如果我们的配置使用的注解方式,那么这里,我的Mapper是没有解析的,当没有解析时,这里就会去解析注解并添加到Configuration(全局配置类)中。
2 注解方式
<bean name="mapperScannerConfigurer_one" class="org.mybatis.spring.mapper.MapperScannerConfigurer"> <property name="sqlSessionFactoryBeanName" value="sqlSessionFactory" /> <property name="basePackage" value="cn.vansky.schedule.time" /> <property name="annotationClass" value="cn.vansky.framework.core.orm.mybatis.annotation.SqlMapper" /> <property name="markerInterface" value="org.mybatis.spring.mapper.MapperFactoryBean" /> </bean>
这里使用注解方式,把包cn.vansky.schedule.time下面的所有JAVA类全部实例化,但是这样实例的话,那么实际类型就是对应JAVA类的类型,也就相当于下面的配置。
<bean id="menuMapper" class="cn.vansky.schedule.time.menu.dao.MenuMapper"> </bean>
这里如果要是直接使用id注入,那么就调用不到MyBatis的配置,所以分析MapperScannerConfigurer类做了哪些操作?
public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry beanDefinitionRegistry) throws BeansException { if (this.processPropertyPlaceHolders) { processPropertyPlaceHolders(); } Scanner scanner = new Scanner(beanDefinitionRegistry); scanner.setResourceLoader(this.applicationContext); scanner.scan(StringUtils.tokenizeToStringArray(this.basePackage, ConfigurableApplicationContext.CONFIG_LOCATION_DELIMITERS)); }
MapperScannerConfigurer类实现BeanDefinitionRegistryPostProcessor,需要执行postProcessBeanDefinitionRegistry方法。上图红框中,Spring把接口MenuMapper类的属性修改了,使此类的实例化最终的Bean变成MapperFactoryBean并注入MapperFactoryBean类的属性mapperInterface。也就相当于XML的配置(查看本章节1)。
3 BEAN注入
JAVA代码里面注入使用注解@Autowired,此注解默认使用类型获取对应的对象。
这里类型MenuMapper,而实际类型为MapperFactoryBean,应该会出现类型转换异常呀!
MyBatis做了进一步处理,MapperFactoryBean实现FactoryBean,必须实现getObject方法。
这里是获取最终的对象,mapperInterface属性就是实际的类型MenuMapper。
SqlSession是SqlSessionTemplate,SqlSessionTemplate的getMapper方法。
通过代理工厂,实际的对象就通过代理实例化完成。最终的代理类就是MapperProxy。
总结
注解方式就是对XML的整合,把多个Bean实例化整合成一个XML配置,这样省去多余的配置信息,而且还提供了注解方式来配置SQL的Mapper信息。
相关文章推荐
- IOS(swift)-数据存储 · NSKeyedArchiver 归档
- android studio入门
- 9、Swift语言中switch条件语句的基本用法
- Android ListView 分组效果实现
- Android 使用OpenCV的三种方式(Android Studio)
- Android图片加载(1)
- 关于android图片加载框架univser-imageloader使用的一些小技巧记录
- 基于Unity有限状态机框架
- Win 10 UWP开发系列:设置AppBarButton的图标
- 第二波Android的心得即将袭来~
- Android中Activity的生命周期
- ANDROID开发环境
- Android
- Android
- 从“Android群英传”谈开去
- IOS开发基础Object-C(02 )—第一个OC类
- mapper映射文件配置之select、resultMap
- 微信问题汇总
- android自定义view实现水平进度条
- 《JavaScript高级编程》读书笔记——引用类型一:Object和Array