Myabtis源码分析终篇-Mybatis是如何访问数据的?

目录

一、  核心组件Executor 分析

1、Executor 介绍

2、Executor 设计模式

3、Executor 的三个重要组件

二、SQL执行StatementHandler组件介绍

三、 结果集处理ResultHandler介绍


一、
 核心组件
Executor 分析

1、Executor 介绍

在上一篇文章中我们了解到,提到 Sqlsession 的功能都是基于 Executor 来实现的,Executor 是MyBaits 核心接口之一,定义了数据库操作最基本的方法,在其内部遵循 JDBC 规范完成对数据库的访问;Executor 类继承机构如下图所示:

Myabtis源码分析终篇-Mybatis是如何访问数据的?_第1张图片

  • Executor: MyBaits 核心接口之一,定义了数据库操作最基本的方法;
  • CacheingExecutor:使用装饰器模式,对真正提供数据库查询的 Executor 增强了二级缓存的 能 力 ; 二 级 缓 存 初 始 化 位 置 : DefaultSqlSessionFactory.openSessionFromDataSource(ExecutorType, TransactionIsolationLevel, boolean);
  • BaseExecutor:抽象类,实现了 executor 接口的大部分方法,主要提供了缓存管理和事务管理的能力,其他子类需要实现的抽象方法为:doUpdate,doQuery 等方法;
  • BatchExecutor: 批量执行所有更新语句,基于 jdbc 的 batch 操作实现批处理;
  • SimpleExecutor: 默认执行器,每次执行都会创建一个 statement,用完后关闭。;
  • ReuseExecutor: 可重用执行器,将 statement 存入 map 中,操作 map 中的 statement 而不会重复创建 statement;

2、Executor 设计模式

模板模式:一个抽象类公开定义了执行它的方法的方式/模板。它的子类可以按需要重写方法实现,但调用将以抽象类中定义的方式进行。定义一个操作中的算法的骨架,而将一些步骤延迟到子类中。模板方法使得子类可以不改变一个算法的结构即可重定义该算法的某些特定实现;类结构如下:

Myabtis源码分析终篇-Mybatis是如何访问数据的?_第2张图片

AbstractClass 中模板方法 template()定义了功能实现的多个步骤,抽象父类只会对其中几个通用的步骤有实现,而一些可定制化的步骤延迟到子类 ConcreteClass1、ConcreteClass2 中实现,子类只能定制某几个特定步骤的实现,而不能改变算法的结构;

应用场景:遇到由一系列步骤构成的过程需要执行,这个过程从高层次上看是相同的, 但是有些步骤的实现可能不同,这个时候就需要考虑用模板模式了;

MyBatis 的执 行器组件是 使用模板 模式的典型应 用, 其中 BaseExecutor 、BaseStatementHandler 是模板模式的最佳实践;BaseExecutor 执行器抽象类,实现了 executor 接口的大部分方法,主要提供了缓存管理和事务管理的能力,其他子类需要实现的抽象方法为:doUpdate,doQuery 等方法;在 BaseExecutor 中进行一次数据库查询操作的流程如下:

Myabtis源码分析终篇-Mybatis是如何访问数据的?_第3张图片

  如上图所示,doQuery 方法是查询数据的结果的子步骤,doQuery 方法有 SIMPLE、REUSER、BATCH 三种实现,这三种不同的实现是在子类中定义的;

  • SimpleExecutor:默认配置,在 doQuery 方法中使用 PrepareStatement 对象访问数据库, 每次访问都要创建新的 PrepareStatement 对象;
  • ReuseExecutor:在 doQuery 方法中,使用预编译 PrepareStatement 对象访问数据库,访问时,会重用缓存中的 statement 对象;
  • BatchExecutor:在 doQuery 方法中,实现批量执行多条 SQL 语句的能力;

3、Executor 的三个重要组件

通过对 SimpleExecutor doQuery()方法的解读发现,Executor 是个指挥官,它在调度三个小弟工作,这三个小弟分别为:

  • StatementHandler:它的作用是使用数据库的 Statement 或PrepareStatement 执行操作,启承上启下作用;
  • ParameterHandler:对预编译的 SQL 语句进行参数设置,SQL 语句中的的占位符“?” 都对应 BoundSql.parameterMappings 集合中的一个元素,在该对象中记录了对应的参数名称以及该参数的相关属性
  • ResultSetHandler:对数据库返回的结果集(ResultSet)进行封装,返回用户指定的实体类型;

Executor 三个组件内部运作流程如下图所示:

Myabtis源码分析终篇-Mybatis是如何访问数据的?_第4张图片

二、SQL执行StatementHandler组件介绍

StatementHandler 完成 Mybatis 最核心的工作,也是 Executor 实现的基础;功能包括:创建statement 对象,为 sql 语句绑定参数,执行增删改查等 SQL 语句、将结果映射集进行转化; StatementHandler 的类继承关系如下图所示:

Myabtis源码分析终篇-Mybatis是如何访问数据的?_第5张图片

  • BaseStatementHandler:所有子类的抽象父类,定义了初始化 statement 的操作顺序,由子类实现具体的实例化不同的 statement(模板模式);
  • RoutingStatementHandler:Excutor 组件真正实例化的子类,使用静态代理模式,根据上下文决定创建哪个具体实体类;
  1. RoutingStatementHandler 是在 Configuration 的 newStatementHandler 中创建的,见下图:Myabtis源码分析终篇-Mybatis是如何访问数据的?_第6张图片
  2. RoutingStatementHandler 中使用动态代理的方式进行请求转发,在构造方法中, 根据上下文(配置)决定创建具体实现类;如下图:Myabtis源码分析终篇-Mybatis是如何访问数据的?_第7张图片
  • SimpleStatmentHandler :使用 statement 对象访问数据库,无须参数化;
  • PreparedStatmentHandler :使用预编译 PrepareStatement 对象访问数据库;
  • CallableStatmentHandler :调用存储过程;

三、 结果集处理ResultHandler介绍

ResultSetHandler 将从数据库查询得到的结果按照映射配置文件的映射规则,映射成相应的结果集对象;在 ResultSetHandler 内部实际是做三个步骤:找到映射匹配规则 → 反射实例化目标对象 → 根据规则填充属性值,为完成这三部ResultHandler 内部调用的流程图如下:

Myabtis源码分析终篇-Mybatis是如何访问数据的?_第8张图片

至此,整个mybatis常用使用以及源码分析就全部介绍完了,希望大家看完之后有所了解,也欢迎大家和我沟通一起学习交流。

你可能感兴趣的:(mybatis,设计模式,1024程序员节,mybatis源码,模板模式,mybatis数据访问,JAVA)