SpringBoot整合Mybatis原理分析
- Mybatis核心概念
- Mybatis编程式用例
- 配置扫描解析过程
- 获得Mapper对象
- 执行sql
Mybatis核心概念
- Configuration 管理mysql-config.xml全局配置关系类
- SqlSessionFactory Session管理工厂接口
- Session SqlSession是一个面向用户(程序员)的接口。SqlSession中提供了很多操作数据库的方法
- Executor 执行器是一个接口(基本执行器、缓存执行器)作用:SqlSession 内部通过执行器操作数据库
- MappedStatement 底层封装对象作用:对操作数据库存储封装,包括 sql 语句、输入输出参数
- StatementHandler 具体操作数据库相关的 handler 接口
Mybatis编程式用例
public void testSelect() throws IOException {
String resource = "mybatis-config.xml";
InputStream inputStream = Resources.getResourceAsStream(resource);
SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
SqlSession session = sqlSessionFactory.openSession(); // ExecutorType.BATCH
try {
BlogMapper mapper = session.getMapper(BlogMapper.class);
Blog blog = mapper.selectBlogById(1);
System.out.println(blog);
} finally {
session.close();
}
}
配置扫描解析过程
- springboot使用自动装配原理注入SqlSessionFactory,对应方法路径为org.mybatis.spring.boot.autoconfigure.MybatisAutoConfiguration#sqlSessionFactory。如果使用的是mybatis-config.xml进行配置,那么在上述方法中会使用XMLConfigBuilder对mybatis-config.xml文件进行解析,并且会把mybatis-config.xml文件中配置的一个个mapper bean生成相应的mapperProxyFactory并且放入mapperRegistry中.MapperRegistry里面维护的其实是一个Map容器,存储接口和代理工厂的映射关系。
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
@Documented
@Import(MapperScannerRegistrar.class)
@Repeatable(MapperScans.class)
public @interface MapperScan
@MapperScan是一个复合注解,其中@Import注解便是起到了扫描注册第三方组件到springIOC的作用。MapperScannerRegistrar类实现了ImportBeanDefinitionRegistrar接口,实现了registerBeanDefinitions方法,主要功能用于动态的注册某一些具有相同特征的一批类到Spring IoC
3.
@Override
public Set<BeanDefinitionHolder> doScan(String... basePackages) {
Set<BeanDefinitionHolder> beanDefinitions = super.doScan(basePackages);
if (beanDefinitions.isEmpty()) {
LOGGER.warn(() -> "No MyBatis mapper was found in '" + Arrays.toString(basePackages) + "' package. Please check your configuration.");
} else {
processBeanDefinitions(beanDefinitions);
}
return beanDefinitions;
}
}
- super.doScan()方法扫描@MapperScan注解中对应的Mapper路径,把该路径下所有符合候选条件的接口mapper放入ioc容器中。
- processBeanDefinitions(beanDefinitions);该方法在doScan()之后进行调用,将容器中mapper bean的真正类型修改为MapperFactoryBean类型。
private void processBeanDefinitions(Set<BeanDefinitionHolder> beanDefinitions) {
GenericBeanDefinition definition;
for (BeanDefinitionHolder holder : beanDefinitions) {
definition = (GenericBeanDefinition) holder.getBeanDefinition();
String beanClassName = definition.getBeanClassName();
// the mapper interface is the original class of the bean
// but, the actual class of the bean is MapperFactoryBean
definition.getConstructorArgumentValues().addGenericArgumentValue(beanClassName); // 构造函数传参,传入初始mapper类型
definition.setBeanClass(this.mapperFactoryBeanClass);
}
获得Mapper对象
public class MapperFactoryBean<T> extends SqlSessionDaoSupport implements FactoryBean<T>
- SqlSessionDaoSupport内部持有SqlSession的实现类SqlSessionTemplate;
- SqlSessionDaoSupport类重写了checkDaoConfig()方法,这个方法主要用于非mybatis-config.xml方式来注册mapper的情况,这个方法会确保把所有的mapper bean生成相应的mapperProxyFactory并且放入mapperRegistry中。调用时机:在Bean的属性值设置完成时被调用(InitializingBean接口)
- 上述说到所有的mapper接口在放入spring容器之后,真正的bean类型被修改为MapperFactoryBean,而MapperFactoryBean实现了FactoryBean接口,所以程序中从ioc容器在获取对应的Mapper实例的话,走的是getObject()方法,最终调用的又是org.apache.ibatis.binding.MapperRegistry#getMapper方法;
public <T> T getMapper(Class<T> type, SqlSession sqlSession) {
final MapperProxyFactory<T> mapperProxyFactory = (MapperProxyFactory<T>) knownMappers.get(type);
if (mapperProxyFactory == null) {
throw new BindingException("Type " + type + " is not known to the MapperRegistry.");
}
try {
return mapperProxyFactory.newInstance(sqlSession);
} catch (Exception e) {
throw new BindingException("Error getting mapper instance. Cause: " + e, e);
}
}
public T newInstance(SqlSession sqlSession) {
final MapperProxy<T> mapperProxy = new MapperProxy<>(sqlSession, mapperInterface, methodCache);
return newInstance(mapperProxy);
}
protected T newInstance(MapperProxy<T> mapperProxy) {
return (T) Proxy.newProxyInstance(mapperInterface.getClassLoader(), new Class[] { mapperInterface }, mapperProxy);
}
- 可以看到获取Mapper的代理对象,实际上是从map中获取对应的工厂类后,最终通过JDK动态代理创建的。实质上是获取了一个MapperProxy的代理对象,MapperProxy中有SqlSession、mapperInterface、methodCache。
执行sql
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
try {
if (Object.class.equals(method.getDeclaringClass())) {
return method.invoke(this, args);
} else if (isDefaultMethod(method)) {
return invokeDefaultMethod(proxy, method, args);
}
} catch (Throwable t) {
throw ExceptionUtil.unwrapThrowable(t);
}
final MapperMethod mapperMethod = cachedMapperMethod(method);
return mapperMethod.execute(sqlSession, args);
}
- 由于所有的Mapper都是MapperProxy的代理对象,所以任意的方法都是执行MapperProxy的invoke()方法。
- 通过Executor、MappedStatement、SqlSession进行sql语句处理,参数填充等等,最终还是通过JDBC包中的PreparedStatement、ResultSet进行最终的调用和结果集处理;