深度解析MyBatis核心:探寻其核心对象的精妙设计

本文涉及的产品
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: 深度解析MyBatis核心:探寻其核心对象的精妙设计

数据存储类对象

Configuration

Configuration 类是 MyBatis 框架的核心配置类,它负责管理 MyBatis 的各种配置信息。在 MyBatis 框架启动时,会通过 XMLConfigBuilder 或者 Java API 读取配置信息并构建 Configuration 对象。

Configuration 对象中包含了 MyBatis 的各种配置信息,例如:

  • 数据库连接池相关配置,如数据源、连接池大小、连接超时等。
  • Mapper 文件路径和 Mapper 接口类的映射关系。
  • 类型处理器(TypeHandler)相关配置,用于将 Java 类型转换为数据库类型或者将数据库类型转换为 Java 类型。
  • 全局 SQL 过滤器、插件等扩展功能的配置。
  • 缓存相关配置,包括一级缓存和二级缓存。
  • 全局属性配置,如默认的查询超时时间、JDBC 驱动类名等。

Configuration 类提供了一系列方法,用于获取和设置各种配置信息。例如,我们可以通过 getDataSource() 方法获取数据源对象,通过 getTypeHandlerRegistry() 方法获取类型处理器注册表对象,通过 addMapper() 方法添加 Mapper 接口等。

Configuration 类还可以创建其他的核心类实例,提供了如 newParameterHandler、newResultSetHandler、newStatementHandler、newExecutor 等方法。

在 MyBatis 框架的整个生命周期中,Configuration 对象是不可变的,也就是说,一旦配置信息被加载到 Configuration 对象中之后,就不能再进行修改。这是因为 MyBatis 框架的设计目标之一就是保持线程安全和可重入性,避免多线程环境下的竞争和锁等问题。

Configuration 是 MyBatis 的核心配置类,封装了 mybatis-config.xml 中的各个配置项。管理所有的 MappedStatement,并且可以通过 Configuration 来创建其他的核心类实例,如 Executor、StatementHandler …

MappedStatement

MappedStatement 是 MyBatis 框架中的一个重要的类,用于描述一个 SQL 语句的详细信息。它是 MyBatis 框架中的一个核心组件,位于 org.apache.ibatis.mapping 包下。

在 MyBatis 框架中,MappedStatement 包含了一个 SQL 语句的所有信息,如 statementId、statementType、resultSetType、参数映射、返回值映射、SQL 语句模板等。MappedStatement 还包含了执行该 SQL 语句所需的其他信息,如缓存配置、动态 SQL 解析器等。

MappedStatement 类的具体属性和用法如下:

  • statementId:SQL 语句的唯一标识符,由 namespace 和 id 组成。
  • statementType:Statement 的类型,STATEMENT、PREPARE、CALLABLE,默认 PREPARE。
  • sqlCommandType:SQL 语句的类型,包括 UNKNOWN、SELECT、INSERT、UPDATE、DELETE、FLUSH 六种类型。
  • resultSetType:SQL 语句的结果集类型,可选值为 FORWARD_ONLY、SCROLL_INSENSITIVE、SCROLL_SENSITIVE 和 DEFAULT。
  • parameterMap:参数映射,用于指定 SQL 语句中的参数类型和参数名称。
  • resultMaps:返回值映射,用于指定返回结果的映射关系。
  • cache:缓存配置,用于配置 SQL 语句的缓存策略。
  • keyGenerator:主键生成策略,用于生成插入操作的主键。
  • timeout:SQL 语句的超时时间,单位为秒。
  • fetchSize:每次从数据库中获取的记录数。
  • statement:SQL 语句模板,可以使用参数占位符(?)表示参数。
  • configuration:关联的核心配置类实例。
  • sqlSource:通过 BoundSql 封装 SQL 语句以及对应的参数。

Mybatis 通过解析 XML,生成 SQL 对应的 MappedStatement,并放入 SqlSessionTemplate 中 Configuration 类属性中,等正真执行 Mapper 接口中的方法时,根据 Mapper 接口的 全类名 + 方法名 作为 Key,会从 Configuration 中找到对应的 MappedStatement,然后进行后续的操作。

Mapper 的 XML 文件中,一个 SQL 语句对应一个 MappedStatement,唯一标识为:namespace.id,MappedStatement 和 Configuration 是双向依赖的,既可以通过 Configuration 找到所有的 MappedStatement,也可以通过 MappedStatement 找到对应的 Configuration。MappedStatement 将 XML 中真正的 SQL 语句和参数封装在了 BoundSql 对象中。

操作类对象

Executor

Executor 类是 MyBatis 框架中的一个核心组件,用于执行 SQL 语句并处理结果。它位于 org.apache.ibatis.executor 包下。

在 MyBatis 中,Executor 负责管理 SQL 语句的执行过程,并将结果映射到相应的对象中。它通常与 StatementHandler、ParameterHandler 和 ResultSetHandler 等其他组件配合工作,完成 SQL 语句的预编译、参数设置、结果集处理等操作。

Executor 类的主要作用可以总结为以下几点:

  1. 执行 SQL 语句:Executor 的核心功能是执行 SQL 语句。它通过调用 StatementHandler 的 prepare、parameterize 和 execute 方法来实现。prepare 方法用于创建 PreparedStatement 对象,parameterize 方法用于设置参数值,execute 方法用于执行 SQL 语句。
  2. 缓存支持:Executor 可以根据配置进行缓存操作,提高 SQL 语句的执行效率。在执行 SQL 语句之前,Executor 会先检查是否存在缓存的结果,如果存在,则直接返回缓存的结果,避免了对数据库的重复访问。
  3. 事务管理:Executor 在执行 SQL 语句时,可以根据配置来管理事务。MyBatis 支持多种事务管理方式,如 JDBC 的自动提交、Spring 事务管理等。Executor 可以在适当的时候开启、提交或回滚事务,确保数据的一致性和完整性。
  4. 结果映射:Executor 通过调用 ResultSetHandler 的 handleResultSets 方法,将 SQL 查询的结果集映射到 Java 对象中。ResultSetHandler 负责将数据库返回的结果转化为 Java 对象,使得开发者可以方便地操作和处理查询结果。
  5. 批处理支持:Executor 提供了对批处理的支持,可以将多个 SQL 语句放在一个批次中执行。这种方式可以减少与数据库的交互次数,提高系统的性能。

Executor 类的具体实现有多种,包括 SimpleExecutor、ReuseExecutor 和 BatchExecutor 等。

  1. SimpleExecutor:SimpleExecutor 是 Executor 接口的默认实现类。它通过 JDBC 的 Statement 对象执行 SQL 语句,每次执行都会创建一个新的 Statement 对象。SimpleExecutor 在执行 SQL 语句时不会使用预编译的 PreparedStatement,而是直接将 SQL 字符串传递给数据库进行执行。这种实现方式适用于简单的场景,不支持缓存和批处理。
  2. ReuseExecutor:ReuseExecutor 是 Executor 接口的另一个实现类,它与 SimpleExecutor 类似,也使用 JDBC 的 Statement 对象执行 SQL 语句。不同之处在于 ReuseExecutor 会重用已经创建的 Statement 对象,而不是每次执行都创建新的对象。这样可以减少 Statement 对象的创建和销毁开销,提高性能。ReuseExecutor 适用于需要频繁执行相同 SQL 语句的场景。
  3. BatchExecutor:BatchExecutor 是 Executor 接口的批处理实现类。它可以将多个 SQL 语句放在一个批次中执行,减少与数据库的交互次数,提高性能。BatchExecutor 在执行 SQL 语句时会使用 JDBC 的 addBatch 和 executeBatch 方法。在执行过程中,它会将所有的 SQL 语句收集起来,然后一次性发送给数据库执行。BatchExecutor 适用于需要批量处理大量数据的场景。

这些实现类都实现了 Executor 接口,因此可以通过配置文件中的 标签来指定使用的实现类。例如:

<configuration>
  <environments default="development">
    <environment id="development">
      <transactionManager type="JDBC"/>
      <executor type="org.apache.ibatis.executor.SimpleExecutor"/>
      <!-- 其他配置 -->
    </environment>
  </environments>
</configuration>

Executor 是一个接口,其中定义了有关增删改(update)、查询(query)、事务(rollback、getTransaction …)、缓存(isCached、clearLocalCache)等方法。他是 MyBatis 中处理功能的核心接口。

StatementHandler

StatementHandler 类是 MyBatis 中用来处理 SQL 语句的核心组件之一,主要负责对 SQL 语句进行解析、参数设置和执行操作。在 MyBatis 中,每个 SQL 操作都会对应一个 StatementHandler 对象。

StatementHandler 接口的实现类有三种类型:SimpleStatementHandler、PreparedStatementHandler(默认) 和 CallableStatementHandler。这三个实现类分别对应着 JDBC 中的 Statement、PreparedStatement 和 CallableStatement。

StatementHandler 接口中定义了以下几个方法:

  • prepare:该方法用于创建 PreparedStatement 或 CallableStatement 对象,并对占位符进行设置。
  • parameterize:该方法用于设置 SQL 语句中的参数。
  • batch:该方法用于执行批处理操作。
  • update:该方法用于执行更新操作,返回受影响的行数。
  • query:该方法用于执行查询操作,返回查询结果集。

默认情况下,MyBatis 使用 RoutingStatementHandler 类作为 StatementHandler 的实现类,它是一个路由器,根据 SQL 语句的类型和配置文件中的参数自动选择合适的 StatementHandler 实现类,这是一个典型的装饰器设计模式。

如果需要修改 StatementHandler 的默认行为,可以通过实现 org.apache.ibatis.executor.statement.StatementHandler 接口,并在配置文件中指定新的 StatementHandler 实现类来实现自定义的 StatementHandler。

真正与 JDBC 产生联系,封装了 JDBC 中的三种 Statement 来完成增删查改操作。

ParameterHandler

ParameterHandler 是 MyBatis 框架中的一个接口,用于处理 SQL 语句中的参数。它负责将 Java 对象中的属性值设置到 PreparedStatement 对象中。

ParameterHandler 接口定义了以下方法:

  • void setParameters(PreparedStatement ps):将 Java 对象中的属性值设置到 PreparedStatement 对象中。

具体来说,ParameterHandler 的实现类需要完成以下工作:

  1. 根据 SQL 语句中的占位符数量,确定参数的个数。
  2. 根据参数的类型和配置的 TypeHandler,将 Java 对象中的属性值转换为对应的数据库类型,并设置到 PreparedStatement 对象中。

ParameterHandler 只有一个实现类 DefaultParameterHandler。

将 MyBatis 中的 Java 参数转变为 JDBC的参数,即完成 @Param -> #{} -> ? 的转变。

public class DefaultParameterHandler implements ParameterHandler {
  // 属性
  // 持有 typeHandler 注册器
  private final TypeHandlerRegistry typeHandlerRegistry;
  // 持有 MappedStatement 实例,这是一个静态的 xml 的一个数据库操作节点的静态信息而已
  private final MappedStatement mappedStatement;
  
  // 持有当前操作传入的实际参数
  private final Object parameterObject;
  // 动态语言被执行后的结果 sql
  private final BoundSql boundSql;
  private final Configuration configuration;
  
  // 构造函数
  public DefaultParameterHandler(MappedStatement mappedStatement, Object parameterObject, BoundSql boundSql) {
    this.mappedStatement = mappedStatement;
    this.configuration = mappedStatement.getConfiguration();
    this.typeHandlerRegistry = mappedStatement.getConfiguration().getTypeHandlerRegistry();
    this.parameterObject = parameterObject;
    this.boundSql = boundSql;
  }
  
  // 实现方法
  @Override
  public Object getParameterObject() {
    return parameterObject;
  }
  
  @Override
  public void setParameters(PreparedStatement ps) {
    ErrorContext.instance().activity("setting parameters").object(mappedStatement.getParameterMap().getId());
    // 1. 获取 boundSql 中的参数映射信息列表
    List<ParameterMapping> parameterMappings = boundSql.getParameterMappings();
    if (parameterMappings != null) {
      // 1.1. 遍历参数映射列表,这个列表信息就是我们 xml 文件中定义的某个查询语句的所有参数映射信息,注意这个 List 中的参数映射元素的顺序是和真实 xml 中 sql 的参数顺序对应的
      for (int i = 0; i < parameterMappings.size(); i++) {
        ParameterMapping parameterMapping = parameterMappings.get(i);
        // 1.2. 只有入参类型才会设置 PreparedStatement
        if (parameterMapping.getMode() != ParameterMode.OUT) {
          Object value;
          // 取出参数名,这里比如说是 'phone'
          String propertyName = parameterMapping.getProperty();
          if (boundSql.hasAdditionalParameter(propertyName)) { // issue #448 ask first for additional params
            value = boundSql.getAdditionalParameter(propertyName);
          } else if (parameterObject == null) {
            value = null;
          } else if (typeHandlerRegistry.hasTypeHandler(parameterObject.getClass())) {
            value = parameterObject;
          } else {
            // 1.3. 这一步的工作就是从当前实际传入的参数中获取到指定 key('phone')的 value 值,比如是'15800000000'
            MetaObject metaObject = configuration.newMetaObject(parameterObject);
            value = metaObject.getValue(propertyName);
          }
          
          // 2. 获取该参数对应的 typeHandler
          TypeHandler typeHandler = parameterMapping.getTypeHandler();
          
          // 2.1. 获取该参数对应的 jdbcType
          JdbcType jdbcType = parameterMapping.getJdbcType();
          if (value == null && jdbcType == null) {
            jdbcType = configuration.getJdbcTypeForNull();
          }
          try {
            // 3. 重点是调用每个参数对应的 typeHandler 的 setParameter 方法为该 ps 设置正确的参数值
            typeHandler.setParameter(ps, i + 1, value, jdbcType);
          } catch (TypeException e) {
            throw new TypeException("Could not set parameters for mapping: " + parameterMapping + ". Cause: " + e, e);
          } catch (SQLException e) {
            throw new TypeException("Could not set parameters for mapping: " + parameterMapping + ". Cause: " + e, e);
          }
        }
      }
    }
  }
}

ResultSetHandler

ResultSetHandler 是 Mybatis 的核心组件,主要负责将结果集 resultSets 转化成结果列表(或 cursor)和处理储存过程的输出。

在原生 JDBC 查询的代码中,使用 Statement 进行操作,会返回 ResultSet 对象。ResultSet 也是 java.sql 中的接口,它表示通过执行查询数据库的语句生成的结果集对象,在 JDBC 的操作中数据库的所有查询记录将使用 ResultSet 进行接收。

我们获取到 ResultSet 后,就可以将其转换为程序中的 Java 对象进行数据展示,但是原生的操作非常繁琐,所以 Mybatis 提供了 ResultSet 处理器,我们只需要定义好返回类型,Mybatis 就可以自动进行转换映射了,底层使用反射技术。

DefaultResultSetHandler 是 ResultSetHandler 的默认实现类,其中实现了很多处理一对一、一对多、嵌套查询等结果集的处理方法。

TypeHandler

TypeHandler 是 MyBatis 中的一个接口,用于处理 Java 对象与数据库数据类型之间的转换。它提供了一种机制,允许你自定义数据类型在数据库和Java对象之间的映射规则。MyBatis 默认提供了一些常见数据类型的 TypeHandler 实现,但是在特定场景下,你可能需要自定义 TypeHandler 来满足你的需求。

类型处理器这个接口其实很简单,总共四个方法,一个方法将入参的 Java 类型的数据转换为 JDBC 类型,三个方法将返回结果转换为 Java 类型。源码如下:

public interface TypeHandler<T> {
  // Java 类型的数据转换为 JDBC 类型
  void setParameter(PreparedStatement ps, int i, T parameter, JdbcType jdbcType) throws SQLException;
  
  // 将返回结果转换为 Java 类型
  T getResult(ResultSet rs, String columnName) throws SQLException;
 
  T getResult(ResultSet rs, int columnIndex) throws SQLException;
 
  T getResult(CallableStatement cs, int columnIndex) throws SQLException;
}

自定义类型处理器的方法有两种,一种是实现 TypeHandler 这个接口,另一个就是继承 BaseTypeHandler 这个便捷的抽象类。实现其中的方法后将类型处理器注入给 MyBatis。可以使用注解如:@MappedTypes@MappedJdbcTypes,也可以在配置文件中指定 mybatis.type-handlers-package=自定义类型处理器的路径


相关文章
|
4月前
|
Java 数据库连接 数据库
mybatis查询数据,返回的对象少了一个字段
mybatis查询数据,返回的对象少了一个字段
331 8
|
2月前
|
SQL Java 数据库连接
canal-starter 监听解析 storeValue 不一样,同样的sql 一个在mybatis执行 一个在数据库操作,导致解析不出正确对象
canal-starter 监听解析 storeValue 不一样,同样的sql 一个在mybatis执行 一个在数据库操作,导致解析不出正确对象
|
3月前
|
存储 编译器 C语言
C++类与对象深度解析(一):从抽象到实践的全面入门指南
C++类与对象深度解析(一):从抽象到实践的全面入门指南
61 8
|
3月前
|
Python
深入解析 Python 中的对象创建与初始化:__new__ 与 __init__ 方法
深入解析 Python 中的对象创建与初始化:__new__ 与 __init__ 方法
30 1
|
5月前
|
安全 Java 数据库连接
后端框架的学习----mybatis框架(3、配置解析)
这篇文章详细介绍了MyBatis框架的核心配置文件解析,包括环境配置、属性配置、类型别名设置、映射器注册以及SqlSessionFactory和SqlSession的生命周期和作用域管理。
后端框架的学习----mybatis框架(3、配置解析)
|
4月前
|
设计模式 存储 人工智能
深度解析Unity游戏开发:从零构建可扩展与可维护的游戏架构,让你的游戏项目在模块化设计、脚本对象运用及状态模式处理中焕发新生,实现高效迭代与团队协作的完美平衡之路
【9月更文挑战第1天】游戏开发中的架构设计是项目成功的关键。良好的架构能提升开发效率并确保项目的长期可维护性和可扩展性。在使用Unity引擎时,合理的架构尤为重要。本文探讨了如何在Unity中实现可扩展且易维护的游戏架构,包括模块化设计、使用脚本对象管理数据、应用设计模式(如状态模式)及采用MVC/MVVM架构模式。通过这些方法,可以显著提高开发效率和游戏质量。例如,模块化设计将游戏拆分为独立模块。
249 3
|
4月前
|
JavaScript 前端开发 API
Javaweb之javascript的BOM对象的详细解析
BOM为Web开发提供了强大的API,允许开发者与浏览器进行深入的交互。合理使用BOM中的对象和方法,可以极大地增强Web应用的功能性和用户体验。需要注意的是,BOM的某些特征可能会在不同浏览器中表现不一致,因此在开发过程中需要进行仔细的测试和兼容性处理。通过掌握BOM,开发者能够制作出更丰富、更动态、更交互性的JavaWeb应用。
45 1
|
5月前
|
存储 JavaScript 前端开发
一文带你深度解析:JavaScript中对象与数组的威力究竟有多大?
一文带你深度解析:JavaScript中对象与数组的威力究竟有多大?
|
2月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
103 2
|
20天前
|
存储 设计模式 算法
【23种设计模式·全精解析 | 行为型模式篇】11种行为型模式的结构概述、案例实现、优缺点、扩展对比、使用场景、源码解析
行为型模式用于描述程序在运行时复杂的流程控制,即描述多个类或对象之间怎样相互协作共同完成单个对象都无法单独完成的任务,它涉及算法与对象间职责的分配。行为型模式分为类行为模式和对象行为模式,前者采用继承机制来在类间分派行为,后者采用组合或聚合在对象间分配行为。由于组合关系或聚合关系比继承关系耦合度低,满足“合成复用原则”,所以对象行为模式比类行为模式具有更大的灵活性。 行为型模式分为: • 模板方法模式 • 策略模式 • 命令模式 • 职责链模式 • 状态模式 • 观察者模式 • 中介者模式 • 迭代器模式 • 访问者模式 • 备忘录模式 • 解释器模式
【23种设计模式·全精解析 | 行为型模式篇】11种行为型模式的结构概述、案例实现、优缺点、扩展对比、使用场景、源码解析

推荐镜像

更多