MyBatis 是一个半自动的持久层框架,它通过 XML 或注解的方式将对象与数据库中的表关联起来,从而简化了 JDBC 的复杂操作。要深入理解 MyBatis 的工作原理,我们需要从其核心组件和执行流程入手,通过源码级别的解读来揭示其背后的机制。
核心组件
-
SqlSessionFactory: 作为 MyBatis 的心脏,
SqlSessionFactory
的实例创建后,可以用来获取SqlSession
。SqlSessionFactoryBuilder
用于构建SqlSessionFactory
实例。 -
SqlSession: 代表和数据库的一次会话,用来执行 SQL,获取映射器(Mapper)和管理事务。
-
Executor: 执行器,是 MyBatis 的核心之一,负责 SQL 语句的生成和查询缓存的维护。
-
StatementHandler: 封装了 JDBC Statement 的操作,负责对 JDBC Statement 的操作,如设置参数、执行语句等。
-
ParameterHandler: 负责对用户传入的参数转换成 JDBC Statement 所需要的参数。
-
ResultSetHandler: 负责将 JDBC 返回的 ResultSet 结果集对象转换成需要的数据对象。
-
Mapper: 映射器,基于接口编程,内部通过动态代理实现,将接口方法调用转换为数据库操作。
执行流程
以查询操作为例,深入理解 MyBatis 的工作原理:
-
配置解析: MyBatis 启动时,通过读取配置文件(mybatis-config.xml)和映射文件(*.xml),构建
Configuration
对象。 -
获取 SqlSession: 通过
SqlSessionFactory
获取SqlSession
实例。 -
获取 Mapper: 通过
SqlSession
获取需要的 Mapper 接口的代理对象。 -
执行查询: 调用 Mapper 接口中定义的方法,MyBatis 通过动态代理,将调用转发给
Executor
执行。 -
SQL 解析:
Executor
根据方法名找到映射的 SQL 语句,并通过ParameterHandler
设置 SQL 参数。 -
执行 SQL:
Executor
通过StatementHandler
准备 Statement,执行 SQL 语句,并通过ResultSetHandler
处理结果集。 -
返回结果: 将处理后的结果返回给调用方。
源码解析
以查询操作的核心流程为例,展示部分关键源码:
获取 SqlSession:
SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
SqlSession sqlSession = sqlSessionFactory.openSession();
获取 Mapper 并执行查询:
BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
Blog blog = mapper.selectBlog(101);
在 MyBatis 内部,getMapper
方法实际上是通过 MapperProxyFactory
创建一个动态代理对象,代理对象在调用方法时,会转发给 MapperMethod
执行。
MapperMethod 执行:
public class MapperMethod {
public Object execute(SqlSession sqlSession, Object[] args) {
Object result;
// 根据方法类型(如SELECT、UPDATE)调用不同的SqlSession方法
result = sqlSession.selectOne(command.getName(), args);
return result;
}
}
SqlSession 执行查询:
public class DefaultSqlSession implements SqlSession {
public <T> T selectOne(String statement, Object parameter) {
// Executor 执行查询
return executor.query(ms, wrapCollection(parameter), RowBounds.DEFAULT, Executor.NO_RESULT_HANDLER);
}
}
总结
通过上述源码级别的解读,我们可以看到 MyBatis 的工作原理主要涉及配置解析、SqlSession 的获取、Mapper 的动态代理、SQL 的解析与执行等关键步骤。理解这些核心组件和执行流程,有助于更深入地掌握 MyBatis 的工作机制,从而在实际开发中更加灵活和高效地使用 MyBatis。