MyBatis 架构总览

MyBatis 架构分为三层,分别是基础支撑层核心处理层接口层,如下图所示:

MyBatis 架构

1 基础支撑层

基础支撑层是整个 MyBatis 框架的地基,为整个 MyBatis 框架提供了非常基础的功能,其中每个模块都提供了一个内聚的、单一的能力,MyBatis 基础支撑层按照这些单一的能力可以划分为上图所示的九个基础模块。

1.1 1. 资源加载模块

该模块通过 I/O 流将配置文件加载到内存,比较简单这里略过了。

1.2 2. 类型转换模块

MyBatis 中 JDBC 类型与 Java 类型之间的相互转换就是通过类型转换模块实现的,此外该模块还实现了 MyBatis 的别名机制 typeAliases

JDBC 与 Java 类型互转这一功能在绑定实参、映射结果集场景中都有所体现:

  • 在 SQL 模板绑定用户传入实参的场景中,类型转换模块会将 Java 类型数据转换成 JDBC 类型数据;
  • 在将结果集映射成结果对象的时候,类型转换模块会将 JDBC 类型数据转换成 Java 类型数据。

具体情况如下图所示:

MyBatis 类型转换

1.3 3. 日志模块

日志是我们生产实践中排查问题、定位 Bug、锁定性能瓶颈的主要线索来源,在任何一个成熟系统中都会有级别合理、信息翔实的日志模块,MyBatis 也不例外。MyBatis 提供了日志模块来集成 Java 生态中的第三方日志框架,该模块目前可以集成 Log4j、Log4j2、slf4j 等优秀的日志框架。

1.4 4. 反射工具模块

Java 中的反射功能非常强大,许多开源框架都会依赖反射实现一些相对灵活的需求,但是大多数 Java 程序员在实际工作中很少会直接使用到反射技术。MyBatis 的反射工具箱是在 Java 反射的基础之上进行的一层封装,为上层使用方提供更加灵活、方便的 API 接口,同时缓存 Java 的原生反射相关的元数据,提升了反射代码执行的效率,优化了反射操作的性能。

1.5 5. Binding 模块

我们可以通过 SqlSession 获取 Mapper 接口的代理,然后通过这个代理执行关联 mapper.xml 文件中的数据库操作。通过这种方式,可以将一些错误提前到编译期,该功能就是通过 Binding 模块完成的。

这里特别说明的是,在使用 MyBatis 的时候,我们无须编写 Mapper 接口的具体实现,而是利用 Binding 模块自动生成 Mapper 接口的动态代理对象。有些简单的数据操作,我们还可以直接在 Mapper 接口中使用注解完成,连 mapper.xml 配置文件都无须编写,但如果 ResultSet 映射以及动态 SQL 非常复杂,还是建议在 mapper.xml 配置文件中维护会比较方便。

1.6 6. 数据源模块

持久层框架核心组件之一就是数据源,一款性能出众的数据源可以成倍提升系统的性能。MyBatis 自身提供了一套不错的数据源实现,也是 MyBatis 的默认实现。另外,在 Java 生态中,就有很多优异开源的数据源可供选择,MyBatis 的数据源模块中也提供了与第三方数据源集成的相关接口,这也为用户提供了更多的选择空间,提升了数据源切换的灵活性。

1.7 7. 缓存模块

数据库是实际生产中非常核心的组件,很多业务数据都会落地到数据库,所以数据库性能的优劣直接影响了上层业务系统的优劣。我们很多线上业务都是读多写少的场景,在数据库遇到瓶颈时,缓存是最有效、最常用的手段之一,正确使用缓存可以将一部分数据库请求拦截在缓存这一层,这就能够减少一部分数据库的压力,提高系统性能。

除了使用 Redis、Memcached 等外置的第三方缓存以外,持久化框架一般也会自带内置的缓存,例如,MyBatis 就提供了一级缓存和二级缓存,具体实现也位于基础支撑层的缓存模块中。

1.8 8. 解析器模块

MyBatis 中有两大配置文件需要解析,一个是 Mybatis 核心配置文件,另一个是 mapper.xml 配置文件。这两个文件都是由 MyBatis 的解析器模块进行解析的,其主要是依赖 XPath 实现 XML 配置文件以及各类表达式的高效解析。

1.9 9. 事务管理模块

持久层框架一般都会提供一套事务管理机制实现数据库的事务控制,MyBatis 对数据库中的事务进行了一层简单的抽象,提供了简单易用的事务接口和实现。一般情况下,Java 项目都会集成 Spring,并由 Spring 框架管理事务。

2 核心处理层

核心处理层是 MyBatis 的核心所在,其中涉及 MyBatis 的初始化以及执行一条 SQL 语句的全流程。

2.1 1. 配置解析

我们知道,MyBatis 有三处可以添加配置信息的地方,分别是:MyBatis 核心配置文件、mapper.xml 配置文件以及 Mapper 接口中的注解信息。在 MyBatis 初始化过程中,会加载这些配置信息,并将解析之后得到的配置对象保存到 Configuration 对象中。

2.2 2. SQL 解析与 scripting 模块

MyBatis 的最大亮点应该要数其动态 SQL 功能了,只需要通过 MyBatis 提供的标签即可根据实际的运行条件动态生成实际执行的 SQL 语句。MyBatis 提供的动态 SQL 标签非常丰富,包括 <where> 标签、<if> 标签、<foreach> 标签、<set> 标签等。

MyBatis 中的 scripting 模块就是负责动态生成 SQL 的核心模块。它会根据运行时用户传入的实参,解析动态 SQL 中的标签,并形成 SQL 模板,然后处理 SQL 模板中的占位符,用运行时的实参填充占位符,得到数据库真正可执行的 SQL 语句。

2.3 3. SQL 执行

在 MyBatis 中,要执行一条 SQL 语句,会涉及非常多的组件,比较核心的有:ExecutorStatementHandlerParameterHandlerResultSetHandler

其中,Executor 会调用事务管理模块实现事务的相关控制,同时会通过缓存模块管理一级缓存和二级缓存。SQL 语句的真正执行将会由 StatementHandler 实现。StatementHandler 会先依赖 ParameterHandler 进行 SQL 模板的实参绑定,然后由 java.sql.Statement 对象将 SQL 语句以及绑定好的实参传到数据库执行,从数据库中拿到 ResultSet。最后,由 ResultSetHandler 将 ResultSet 映射成 Java 对象返回给调用方,这就是 SQL 执行模块的核心。

2.4 4. 插件

很多成熟的开源框架,都会以各种方式提供扩展能力。当框架原生能力不能满足某些场景的时候,就可以针对这些场景实现一些插件来满足需求,这样的框架才能有足够的生命力。这也是 MyBatis 插件接口存在的意义。

与此同时,在实际应用的时候,我们也可以通过自定义插件来扩展 MyBatis,或者改变 MyBatis 的默认行为。因为插件会影响 MyBatis 内核的行为,所以在自定义插件之前,我们必须非常了解 MyBatis 内部的运行原理,以避免写出不符合预期的插件,引入一些诡异的功能 Bug 或性能问题。

3 接口层

接口层是 MyBatis 暴露给调用的接口集合,这些接口都是使用 MyBatis 时最常用的一些接口,例如,SqlSession 接口、SqlSessionFactory 接口等。其中,最核心的是 SqlSession 接口,我们可以通过它实现很多功能,例如:获取 Mapper 代理、执行 SQL 语句、控制事务开关等。


欢迎关注我的公众号,第一时间获取文章更新:

微信公众号

相关内容