MyBatis中SqlSession创建过程详解
1、SqlSession接口介绍
SqlSession是MyBatis的主要Java接口,通过这个接口可以执行命令、获取mappers、进行事务管理。
接口的主要方法有:
data:image/s3,"s3://crabby-images/717b1/717b1c8dd6bcb0b6eeee93160ee34cc583cefbf3" alt=""
getMapper(Class
实现类:
data:image/s3,"s3://crabby-images/fddc9/fddc9f96bc9b2fe25b8325c687fc799f72282964" alt=""
2、SqlSessionFactory
由SqlSessionFactory
接口创建SqlSession,SqlSessionFactory
有一个唯一的必要属性:用于 JDBC 的 DataSource
。这可以是任意的 DataSource
对象,它的配置方法和其它 Spring 数据库连接是一样的。
在基础的 MyBatis 用法中,是通过 SqlSessionFactoryBuilder
来创建 SqlSessionFactory
的。而在 MyBatis-Spring 中,则使用 SqlSessionFactoryBean
来创建。
示例代码如下:
1 |
|
SqlSessionFactoryBean
实现了Spring的FactoryBean
接口。这意味着由Spring最终创建的bean并不是SqlSessionFactoryBean
本身,而是工厂类(SqlSessionFactoryBean
)的getObject()方法的返回结果。
data:image/s3,"s3://crabby-images/6de1b/6de1b7f6de50092cb6d7cb0d688bb166606faaf7" alt=""
3、追踪mapper的创建过程
通过MapperFactoryBean创建mapper的bean对象
在容器注入MapperSessionFactoryBean
xml方式:
1 | <bean id="userMapper" class="org.mybatis.spring.mapper.MapperFactoryBean"> |
注解方式:
1 |
|
获取mapper
1 |
|
源码解析
MapperFactoryBean类:
继承自SqlSessionDaoSupport
data:image/s3,"s3://crabby-images/eff69/eff69f6ddd0a15ba39ee96903fc52af56ee5c4a1" alt=""
通过构造器将UserMapper
接口的类型赋给mapperInterface属性
data:image/s3,"s3://crabby-images/c3516/c3516c68e92e4e79ab4b4034bbc363aceea8bf41" alt=""
getObject()方法得到UserMapper
接口的代理对象
data:image/s3,"s3://crabby-images/5a6e3/5a6e3ad9b5b8fec48ea9659e95f598d749624dd8" alt=""
getSqlSession()方法是父类SqlSessionDaoSupport
的方法
data:image/s3,"s3://crabby-images/d4290/d4290e51df50fff21a4438fb7280db544f5522e2" alt=""
data:image/s3,"s3://crabby-images/a9502/a95025ecc675f9cf02f20d0b3c4f19da706b8af7" alt=""
由上可知,MapperFactoryBean.getObject()
方法的返回值等同于
new SqlSessionTemplate(sqlSessionFactory).getMapper(mapperInterface)
的返回值
注意:SqlSessionTemplate是SqlSession的实现类
SqlSessionTemplate.getMapper(type)方法
data:image/s3,"s3://crabby-images/a4770/a4770a47c74ecbbde440c95497cd02b393be87e9" alt=""
Configuration.getMapper(type, sqlSession)方法
data:image/s3,"s3://crabby-images/c55c9/c55c962d48d2595987b769a3968e17ad68ee517a" alt=""
MapperRegistry.getMapper(type, sqlSession)方法
data:image/s3,"s3://crabby-images/033b7/033b78d8cafaaefddb2b5effb5f001ea948c0e83" alt=""
MapperProxyFactory.newInstance(sqlSession)方法,Proxy.newProxyInstance()是JDK动态代理的方法
data:image/s3,"s3://crabby-images/4dc57/4dc578737d67e6dd8cc85cc5cbb4a95e2dd3c5c7" alt=""
由此可追踪到,mapper是实际上就是通过动态代理创建的代理对象。
3、mapper执行sql方法
MapperProxy.execute(sqlSession, args)方法
data:image/s3,"s3://crabby-images/0a3b3/0a3b3a5b81696a51992f73ed77a04e497067dc23" alt=""
上图中的sqlSession
为SqlSessionTemplate
,SqlSessionTemplate.insert(statement, parameter)方法:
data:image/s3,"s3://crabby-images/70373/7037351b0e63626c151a1661aa0c06529bcfdbf5" alt=""
SqlSessionProxy的创建,SqlSessionProxy是SqlSession接口的代理对象:
data:image/s3,"s3://crabby-images/9d5b6/9d5b61c18ecb5c645b6b0e63d17dcecb3f14d52b" alt=""
SqlSessionTemplate.SqlSessionInterceptor.invoke()方法中完成事务事务管理