当前位置: 首页 > news >正文

从仿写持久层框架到MyBatis核心源码阅读

接上篇手写持久层框架:https://blog.csdn.net/liwenyang1992/article/details/134884703

MyBatis源码

MyBatis架构原理&主要组件

MyBatis架构设计

在这里插入图片描述

MyBatis架构四层作用是什么呢?

API接口层:提供API,增加、删除、修改、查询等接口,通过API接口对数据库进行操作。

数据处理层:主要负责SQL的 查询、解析、执行以及结果映射的处理,主要作用解析SQL根据调用请求完成一次数据库操作。

框架支撑层:负责通用基础服务支撑,包含事务管理、连接池管理、缓存管理等共用组件的封装,为上层提供基础服务支撑。

在这里插入图片描述

包路径org.apache.ibatis

包路径作用备注
annotationsMapper映射器接口中使用到的注解
bindingMapper映射器接口与映射语句关系绑定构建
builderConfiguration配置的构建包
cache缓存实现与定义(包含一级/二级缓存)
cursor游标(针对查询结果集的获取与遍历等)
datasource数据源/连接池
exceptions异常包
executor语句执行器(包含参数/结果集/语句处理等)
io资源读取辅助包
jdbcMyBatis内部的SQL脚本运行的测试包
logging一套日志接口和适配器包
mappingMapper映射器相关参数/语句/结果/类型等对象包
parsingXML解析包(例如#{}占位符解析)
plugin插件包
reflection反射处理工具包
scriptingSQL执行脚本的解析处理包
session数据库连接会话核心包(会话创建/管理/调用)
transaction事务
type类型处理器(定义bean与数据库类型的转换关系)
XML映射器

MyBatis的真正强大在于它的语句映射,这是它的魔力所在,由于它的异常强大,映射器的XML文件就显得相对简单。如果拿它跟具有相同功能的JDBC代码进行对比,你会立即发现省掉了将近95% 的代码。MyBatis致力于减少使用成本,让用户能更专注于 SQL 代码。

SQL 映射文件只有很少的几个顶级元素(按照应被定义的顺序列出):

  • cache:该命名空间的缓存配置
  • cache-ref:引用其它命名空间的缓存配置。
  • resultMap:描述如何从数据库结果集中加载对象,是最复杂也是最强人的元素
  • sql:可被其它语句引用的可重用语句块。
  • insert:映射插入语句。
  • update:映射更新语句
  • delete:映射删除语句。
  • select:映射查询语句。
select

select元素允许你配置很多属性来配置每条语句的行为细节

    <selectid="select"parameterType="int"resultType="resultType"resultMap="resultMap"flushCache="false"useCache="true"timeout="10"fetchSize="256"statementType="PREPARED"resultSetType="FORWARD_ONLY"></select>
insert, update 和 delete

数据变更语句insert, update 和 delete 的实现非常接近

动态SQL

借助功能强大的基于OGNL的表达式,MyBatis 3 替换了之前的大部分元素,大大精简了元素种类

<select id="findActiveBlogLike"resultType="Blog">SELECT * FROM BLOG WHERE state = ‘ACTIVE’<choose><when test="title != null">AND title like #{title}</when><when test="author != null and author.name != null">AND author_name like #{author.name}</when><otherwise>AND featured = 1</otherwise></choose>
</select>

XMLScriptBuilder

public SqlSource parseScriptNode()方法:

  1. 解析select、insert、update、delete标签中的SQL语句
  2. 最终将解析到的SqlNode封装到MixedSqlNode中的List集合中
  • 将带有${}号的SQL信息封装到TextSqlNode
  • 将带有#{}号的SQL信息封装到StaticTextSqlNode
  • 将动态SQL标签中的SQL信息分别封装到不同的SqlNode中
  • 如果SQL中包含${}和动态SQL语句,则将SqlNode封装到DynamicSqlSource

在这里插入图片描述

相关类与接口
DefaultSqlSession

SqlSession接口的默认实现类

Executor接口:

在这里插入图片描述

BaseExecutor:基础执行器,封装了子类的公共方法及公共变量,包括一级缓存、延迟加载、回滚、关闭等功能;

SimpleExecutor:简单执行器,每执行一条SQL,都会打开一个 Statement,执行完成后关闭;

ReuseExecutor:重用执行器,相较于 SimpleExecutor多了 Statement 的缓存功能,其内部维护一个Map<String, Statement>,每次编译完成的Statement 都会进行缓存,不会关闭;

BatchExecutor:批量执行器,基于JDBC的addBatch、executeBatch功能,并且在当前SQL和上一条SQL完全一样的时候,重用Statement,在调用doFlushStatements的时候,将数据刷新到数据库;

CachingExecutor:缓存执行器,装饰器模式,在开启缓存的时候。会在上面三种执行器的外面包上 CachingExecutor;

缓存执行过程

在这里插入图片描述

SimpleExecutor#doQuery

在这里插入图片描述

  • BaseStatementHandler:基础语句处理器(抽象类),它基本把语句处理器接口的核心部分都实现了,包括配置绑定、执行器绑定、映射器绑定、参数处理器构建、结果集处理器构建、语句超时设置、语句关闭等,并另外定义了新的方法 instantiateStatement供不同子类实现以使获取不同类型的语句连接,子类可以普通执行 SQL语句,也可以做预编译执行,还可以执行存储过程等
  • SimpleStatementHandler:普通语句处理器,继承BaseStatementHandler抽象类,对应 java.sql.Statement 对象的外理,处理普通的不带动态参数运行的SQL,即执行简单拼接的字符串语句,同时由于 Statement 的特性,SimpleStatementHandler 每次执行都需要编译 SQL**(注意:我们知道 SQL 的执行是需要编译和解析的)。**
  • PreparedStatementHandler:预编译语句处理器,继承BaseStatementHandler抽象类,对应 java.sql.PrepareStatement 对象的处理,相比上面的普通语句处理器,它支持可变参数 SQL执行,由于 PrepareStatement 的特性,它会进行预编译,在缓存中一旦发现有预编译的命令,会直接解析执行,所以减少了再次编译环节,能够有效提高系统性能,并预防 SQL 注入攻击**(所以是系统默认也是我们推荐的语句处理器)**
  • CallableStatementHandler:存储过程外理器,继承BaseStatementHandler抽象类,对应 java.sql.CallableStatement 对象的处理,很明了,它是用来调用存储过程的,增加了存储过程的函数调用以及输出/输入参数的处理支持。
  • RoutingStatementHandler:路由语句处理器,直接实现了 StatementHandler 接口,作用如其名称,确确实实只是起到了路由功能,并把上面介绍到的三个语句处理器实例作为自身的委托对象而已,所以执行器在构建语句处理器时,都是直接 new了RoutingStatementHandler实例。

附件中包含测试用例代码,可设置jdk17调试运行,带注解。

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • Docker 从入门到实践:Docker介绍
  • 第三百三十九回
  • 2020年认证杯SPSSPRO杯数学建模B题(第一阶段)分布式无线广播全过程文档及程序
  • KVM虚拟机部署K8S重启后/etc/hosts内容丢失
  • 1分钟带你了解golang(go语言)
  • Linux 安装Jupyter notebook 并开启远程访问
  • WPF 基础入门(XAML理解二)
  • 【头歌实训】PySpark Streaming 入门
  • Linux 硬件配置
  • Android : 使用GestureOverlayView进行手势识别—简单应用
  • Python学习笔记(六)面向对象编程
  • PHP调用系统命令/其他应用程序 并获取应用返回值的方法
  • mac 生成 本地.ssh
  • oracle 9i10g编程艺术-读书笔记2
  • python/selenium/jenkins整合
  • php的引用
  • 9月CHINA-PUB-OPENDAY技术沙龙——IPHONE
  • JavaScript-如何实现克隆(clone)函数
  • 【挥舞JS】JS实现继承,封装一个extends方法
  • 2017 前端面试准备 - 收藏集 - 掘金
  • classpath对获取配置文件的影响
  • JavaScript实现分页效果
  • Java读取Properties文件的六种方法
  • JDK 6和JDK 7中的substring()方法
  • Nacos系列:Nacos的Java SDK使用
  • SpiderData 2019年2月25日 DApp数据排行榜
  • 初识 beanstalkd
  • 电商搜索引擎的架构设计和性能优化
  • 翻译 | 老司机带你秒懂内存管理 - 第一部(共三部)
  • 关于springcloud Gateway中的限流
  • 记录:CentOS7.2配置LNMP环境记录
  • 排序算法学习笔记
  • 异常机制详解
  • UI设计初学者应该如何入门?
  • 选择阿里云数据库HBase版十大理由
  • ​secrets --- 生成管理密码的安全随机数​
  • ​一些不规范的GTID使用场景
  • #预处理和函数的对比以及条件编译
  • ( 用例图)定义了系统的功能需求,它是从系统的外部看系统功能,并不描述系统内部对功能的具体实现
  • (2024)docker-compose实战 (9)部署多项目环境(LAMP+react+vue+redis+mysql+nginx)
  • (附源码)springboot人体健康检测微信小程序 毕业设计 012142
  • (附源码)基于SSM多源异构数据关联技术构建智能校园-计算机毕设 64366
  • (亲测成功)在centos7.5上安装kvm,通过VNC远程连接并创建多台ubuntu虚拟机(ubuntu server版本)...
  • (十)c52学习之旅-定时器实验
  • (转)Android中使用ormlite实现持久化(一)--HelloOrmLite
  • (转)我也是一只IT小小鸟
  • (转载)在C#用WM_COPYDATA消息来实现两个进程之间传递数据
  • *++p:p先自+,然后*p,最终为3 ++*p:先*p,即arr[0]=1,然后再++,最终为2 *p++:值为arr[0],即1,该语句执行完毕后,p指向arr[1]
  • .NET Core 中的路径问题
  • .net dataexcel winform控件 更新 日志
  • .net 生成二级域名
  • .net(C#)中String.Format如何使用
  • .net网站发布-允许更新此预编译站点
  • /etc/fstab 只读无法修改的解决办法
  • @JoinTable会自动删除关联表的数据