Mybatis整合Spring实现Mapper自动注入原理
通过Spring的扩展点,实现bean的管理 + 封装Template方法
一、Bean的管理
-
在引入mybatis-spring.jar后,我们需要在applicationContext.xml中配置
SqlSessionFactoryBean
具体看看这个类:1、FactoryBean:这是spring的工厂bean,我们获取bean的时候会得到它getObject方法返回的对象,只有在beanName前加上“$”才会获得FactoryBean本身的对象
2、InitializingBean:实现这个接口的bean在初始化之前会调用afterPropertiesSet方法;还有这种功能的有initMethod(),@PostConstract
-
看看afterPropertiesSet方法都干了啥
buildSqlSessionFactory方法:
最终返回DefaultSqlSessionFactory
二、封装Template方法(类似HirbnateDaoSupport,mybatis继承SqlSessionDaoSupport能获取SqlSessionTemplate)
-
为什么spring只需要注入我们写的mapper类而不需要去获取会话?
-
为什么用SqlSessionTemplate而不使用DefaultSqlSession?
这里为什么线程不安全呢?因为我们同一个事务中每次获取DefaultSqlSession都会新建一个,这样每个connection就不一样,jdbc的connection本来就是线程不安全的,所以mybatis提供了SqlSessionTemplate来给SqlSession代理,保证一个事务中获取的都是同一个代理的DefaultSqlSession(connection就是同一个)
接下来看看Spring怎么实现我们Mapper接口偷梁换柱和控制线程安全
- 第二个核心配置
这里调用mybatis继承了spring的BeanDefinition扫描类拿到mapper接口的BeanDefinition
看看MapperFactoryBean都有些什么:
SqlSessionTemplate创建:
因此当我们注入mapper,实际注入的是MapperFactoryBean实现FactoryBean接口的getObject方法返回的代理对象:
下面我们执行一下查询的流程:
代理mapper的invoke,并把SqlSessionTemplate传入
最终是DefaultSqlSession的代理对象SqlSessionProxy(在上面SqlSessionTemplate初始化时候创建的
)执行invoke方法
里面最关键的两个方法getSqlSession
和closeSqlSession
- 第二个核心配置