Mybatis框架之责任链模式 (Chain of Responsibility Pattern)
在 MyBatis 框架中,责任链模式 (Chain of Responsibility Pattern) 被广泛应用于多个功能模块中,例如 插件拦截器、SQL 执行流程中的拦截器链、动态 SQL 的解析与处理等。这种设计模式为 MyBatis 提供了高度的扩展性和灵活性,使其能够轻松应对各种自定义功能需求。
1. 什么是责任链模式 (Chain of Responsibility Pattern)?
责任链模式 是一种行为设计模式,用于将请求沿着处理者链进行传递,直到有一个处理者能够处理它为止。责任链模式可以动态地添加或删除处理者,从而提高系统的灵活性。
特点:
- 请求沿链传递:请求从链的头开始传递,每个处理者都可以选择处理请求或将其传递到下一个处理者。
- 解耦请求发送者和处理者:请求发送者无需知道处理者的存在,也无需知道请求是如何被处理的。
- 链式调用:通过将多个处理者串联起来,可以实现责任链的灵活组合。
2. MyBatis 中责任链模式的应用场景
MyBatis 中责任链模式的主要应用包括:
- 插件机制 (Interceptor Chain):MyBatis 提供了强大的插件机制,允许用户通过自定义拦截器来扩展 MyBatis 的功能。所有拦截器组成一个责任链,按顺序处理 SQL 执行的各个阶段(如
prepare
、parameterize
、execute
等)。 - 动态 SQL 的解析:MyBatis 通过解析链来处理动态 SQL 语句(如
<if>
、<choose>
、<foreach>
等)。 - 多种执行器链 (Executor Chain):MyBatis 允许多个
Executor
(如SimpleExecutor
、ReuseExecutor
、BatchExecutor
)协同工作来执行 SQL 语句。
3. MyBatis 插件机制中的责任链模式
MyBatis 中的插件机制采用了典型的责任链模式来处理多个插件。用户可以通过实现 Interceptor
接口来创建自定义插件,并将其配置到 MyBatis 中。每个插件都可以选择拦截 Executor
、StatementHandler
、ParameterHandler
、ResultSetHandler
的方法调用。
3.1 插件机制的架构
Interceptor
接口:定义了插件的基本行为。Plugin
类:负责将插件链与 MyBatis 核心对象连接起来。- 拦截器链 (Interceptor Chain):由多个
Interceptor
组成,按顺序执行。
3.2 插件机制的工作流程
- 用户定义一个或多个自定义拦截器,实现
Interceptor
接口。 - MyBatis 在初始化时读取配置文件,将用户定义的拦截器注册到拦截器链中。
- 当执行 SQL 操作时,MyBatis 将请求依次传递给拦截器链中的每个拦截器。
- 每个拦截器可以选择拦截请求,或者将请求传递给下一个拦截器。
4. 代码示例:MyBatis 插件机制的责任链模式
Step 1:定义自定义拦截器
import org.apache.ibatis.plugin.*;import java.util.Properties;@Intercepts({@Signature(type = Executor.class, method = "update", args = {MappedStatement.class, Object.class})})
public class MyCustomInterceptor implements Interceptor {@Overridepublic Object intercept(Invocation invocation) throws Throwable {System.out.println("Before executing SQL update...");Object result = invocation.proceed(); // 传递给下一个拦截器或执行原方法System.out.println("After executing SQL update...");return result;}@Overridepublic Object plugin(Object target) {return Plugin.wrap(target, this);}@Overridepublic void setProperties(Properties properties) {// 可以设置插件的自定义属性}
}
Step 2:配置插件到 MyBatis 配置文件 (mybatis-config.xml
)
<plugins><plugin interceptor="com.example.MyCustomInterceptor"><property name="someProperty" value="someValue" /></plugin>
</plugins>
Step 3:插件执行流程
当执行 update
方法时,MyBatis 会将请求传递给 MyCustomInterceptor
。如果有多个拦截器,它们会按顺序执行,类似于以下伪代码:
InterceptorChain interceptorChain = configuration.getInterceptorChain();
Object result = interceptorChain.pluginAll(executor); // 将所有拦截器应用到 executor
executor.update(...); // 最终执行 SQL 更新操作
5. 拦截器链 (InterceptorChain
) 的源码解析
public class InterceptorChain {private final List<Interceptor> interceptors = new ArrayList<>();// 添加拦截器public void addInterceptor(Interceptor interceptor) {interceptors.add(interceptor);}// 将所有拦截器应用到目标对象public Object pluginAll(Object target) {for (Interceptor interceptor : interceptors) {target = interceptor.plugin(target);}return target;}
}
6. MyBatis 动态 SQL 解析中的责任链模式
MyBatis 处理动态 SQL 时,使用了不同的 SQL 节点解析器(如 IfSqlNode
、ChooseSqlNode
、TrimSqlNode
等)来解析和构建最终的 SQL 语句。这些解析器通过责任链模式协同工作,从而灵活处理复杂的动态 SQL。
示例:动态 SQL 解析
<select id="findUsers" parameterType="map" resultType="User">SELECT * FROM users<where><if test="name != null"> AND name = #{name} </if><if test="email != null"> AND email = #{email} </if></where>
</select>
MyBatis 会将 <if>
标签解析为 IfSqlNode
,并按顺序执行所有 SQL 节点,最终生成完整的 SQL 语句。
7. MyBatis 中责任链模式的优势
- 高度扩展性:通过责任链模式,MyBatis 插件机制允许用户动态添加或删除拦截器,从而实现各种自定义功能。
- 解耦各个处理步骤:责任链模式将请求的处理逻辑分散到多个拦截器或解析器中,使得代码模块化、清晰易读。
- 灵活性:用户可以自由配置多个插件或动态 SQL 解析器,从而增强 MyBatis 的灵活性和功能性。
8. MyBatis 责任链模式的不足
- 调试困难:由于请求在多个拦截器之间传递,如果某个拦截器出错,可能会增加调试难度。
- 性能开销:如果责任链中包含大量拦截器或处理器,可能会对系统性能产生一定的影响。因此,建议合理使用拦截器,以避免过度的链式调用。
9. 总结
MyBatis 中的责任链模式在多个模块中得到了广泛应用,包括插件机制、动态 SQL 解析和执行器链等。通过责任链模式,MyBatis 实现了灵活的扩展机制,使开发者能够轻松自定义 SQL 执行流程,从而增强了 MyBatis 的功能性和可维护性。因此,责任链模式是 MyBatis 框架设计中的一个重要设计模式,使得其在处理复杂业务场景时更加灵活和高效。
相关文章:
Mybatis框架之责任链模式 (Chain of Responsibility Pattern)
在 MyBatis 框架中,责任链模式 (Chain of Responsibility Pattern) 被广泛应用于多个功能模块中,例如 插件拦截器、SQL 执行流程中的拦截器链、动态 SQL 的解析与处理等。这种设计模式为 MyBatis 提供了高度的扩展性和灵活性,使其能够轻松应对…...

C++ Stack和Queue---单向守护与无尽等待:数据结构的诗意表达
公主请阅 容器适配器容器适配器的特点 栈和队列的模拟实现deque的介绍1. 内存开销较高2.随机访问性能略低于 vector3. 与指针或迭代器的兼容性r4. 不适合用于需要频繁中间插入和删除的场景5. 在特定平台上的实现不一致6. 缺乏shrink_to_fit支持总结 题目 priority_queue 优先级…...

深入理解Java包装类与泛型的应用
今天我将带领大家进入Java包装类和泛型应用的学习。 我的个人主页 我的Java-数据结构专栏 :Java-数据结构,希望能帮助到大家。 一、Java包装类基础 二、Java泛型基础 三、Java包装类与泛型的结合 四、Java泛型进阶 五、Java包装类与泛型实战 一、Ja…...
【机器学习chp4】特征工程
推荐文章1,其中详细分析了为什么L1正则化可以实现特征选择(特征剔除) 【王木头 L1、L2正则化】三个角度理解L1、L2正则化的本质-CSDN博客 推荐文章2,里面详细分析了奇异值分解 【线性代数】矩阵变换-CSDN博客 本文遗留问题&#…...

LeetCode螺旋矩阵
快一个月没刷题了,最近工作有些忙,今天闲下来两小时,刷一道 题目描述 给你一个 m 行 n 列的矩阵 matrix ,请按照 顺时针螺旋顺序 ,返回矩阵中的所有元素。 示例 1: 输入:matrix [[1,2,3],[4…...

第十五届蓝桥杯JAVA的B组题目详情解析
(第一个填空太简单,就不写了,根本不用代码,直接excel计算) 目录 蓝桥杯第二个填空,类斐波那契循环数 蓝桥杯JAVA.b组第三题 -分布式队列(模拟) 食堂(蓝桥杯D题) 编辑 星际旅行(Floyd佛洛依德) 其余的有点变态,感觉学了好像…...

在几分钟内将数据从 Oracle 迁移到 ClickHouse
ClickHouse 是一个开源的面向列的数据库管理系统。它在实时数据处理方面的出色性能显着增强了数据分析和业务洞察力。将数据从 Oracle 迁移到 ClickHouse 可以释放数据在决策中的力量,这是单独使用 Oracle 无法实现的。 本教程介绍如何使用 BladePipe 将数据从 Orac…...

ASP.NET MVC宠物商城系统
该系统采用B/S架构,使用C#编程语言进行开发,以ASP.NET MVC框架为基础,以Visual Studio 2019为开发工具,数据库采用SQL Server进行保存数据。系统主要功能包括登录注册、宠物展示、个人中心、我的订单、购物车、用户管理、宠物类别…...

完整http服务器
目录 背景目标描述技术特点开发环境WWW客户端浏览发展史服务端http发展史http分层概览 背景 http协议被广泛使用,从移动端,pc浏览器,http无疑是打开互联网应用窗口的重要协议,http在网络应用层中的地位不可撼动,是能…...

【专题】2024AIGC创新应用洞察报告汇总PDF洞察(附原数据表)
原文链接:https://tecdat.cn/?p38310 在科技日新月异的今天,人工智能领域正以前所未有的速度发展,AIGC(人工智能生成内容)成为其中最耀眼的明珠。从其应用场景的不断拓展,到对各行业的深刻变革࿰…...

形态学图像处理(Morphological Image Processing)
形态学图像处理(Morphological Image Processing) 前言 本博客为个人总结数字图像处理一课所写,并给出适当的扩展和相应的demo。 写博客跟做 checkpoint 很像,毕竟个人还不能达到那种信手拈来的境界,忘了就是从零开始训练࿰…...

【IDER、PyCharm】免费AI编程工具完整教程:ChatGPT Free - Support Key call AI GPT-o1 Claude3.5
文章目录 CodeMoss 简介CodeMoss 的模型集成如何安装和配置 CodeMossIDER 插件安装步骤 CodeMoss 的实战使用AI 问答功能代码优化与解释优化这段代码解释这段代码 文件上传与对话联网查询与 GPT 助手联网查询GPT 助手 提升开发效率的最佳实践结语更多文献 CodeMoss 简介 CodeM…...

C++11的一些实用特性
1.统一的列表初始化 在C98中,标准允许使用花括号{}对数组或者结构体元素进行统一的列表初始值设定。 //统一的列表初始化 struct Date {int year;int month;int day; };void test1() {Date d1 { 2024,11,14 };int array1[] { 1, 2, 3, 4, 5 };int array2[5] { …...

23种设计模式-观察者(Observer)设计模式
文章目录 一.什么是观察者模式?二.观察者模式的结构三.观察者模式的应用场景四.观察者模式的优缺点五.观察者模式的实现(C示例)六.观察者模式的实现(JAVA示例)七.代码解释八.总结 类图: 观察者设计模式类图…...

【CUDA】Branch Divergence and Unrolling Loop
目录 一、避免分支发散 1.1 并行规约问题 1.2 并行规约中的发散 二、UNrolling Loops 一、避免分支发散 控制流有时依赖于 thread 索引。同一个warp中,一个条件分支可能导致性能很差。通过重新组织数据获取模式可以减少或避免 warp divergence。具体问题查看下…...
深度学习:卷积神经网络的计算复杂度,顺序操作,最大路径长度
卷积层的计算复杂度 在深度学习中,卷积层的计算复杂度主要取决于卷积核的大小、输入和输出的通道数量、以及输入序列的长度。具体来说,卷积层的计算复杂度可以通过以下几个因素来计算: 卷积核大小 k:卷积核的大小决定了每次卷积操…...
springboot 配置文件中 multipart.max-file-size 各个版本的写法
由于springboot具有几个版本,不同版本对于文件上传最大限制的配置也有所不同。 所以要注意springboot本身的版本,不然会一直报错 在springboot1.3版本中: multipart.maxFileSize在springboot1.4与springboot1.5版本中: spring…...

linux 中mysql查看慢日志
1、到mysql容器,先登录到数据库,查看是否开启 mysql -h 127.0.0.1 -uroot -p SHOW VARIABLES LIKE slow_query_log; 2、如果没有开启,需要先开启 set global slow_query_log ON; 3、查看慢日志文件 SHOW VARIABLES LIKE slow_query_log…...
单片机的基本组成与工作原理
单片机(Microcontroller Unit, MCU)是一种将计算机的主要部分集成在一个芯片上的小型计算机系统。它通常包括中央处理器(CPU)、存储器(Memory)、输入输出接口(I/O Ports)、定时器/计…...
智慧隧道和智慧交通
通过引入先进的物联网技术,将各种硬件设备如传感器、摄像头、控制系统等有效地连接并管理起来,以实现道路安全和交通流畅的目标。这些设备将能够实时监控和控制隧道内的各种设备和系统,从而提高道路安全、提升驾驶体验并降低管理成本。 在这个…...
pam_env.so模块配置解析
在PAM(Pluggable Authentication Modules)配置中, /etc/pam.d/su 文件相关配置含义如下: 配置解析 auth required pam_env.so1. 字段分解 字段值说明模块类型auth认证类模块,负责验证用户身份&am…...

Mac软件卸载指南,简单易懂!
刚和Adobe分手,它却总在Library里给你写"回忆录"?卸载的Final Cut Pro像电子幽灵般阴魂不散?总是会有残留文件,别慌!这份Mac软件卸载指南,将用最硬核的方式教你"数字分手术"࿰…...
高防服务器能够抵御哪些网络攻击呢?
高防服务器作为一种有着高度防御能力的服务器,可以帮助网站应对分布式拒绝服务攻击,有效识别和清理一些恶意的网络流量,为用户提供安全且稳定的网络环境,那么,高防服务器一般都可以抵御哪些网络攻击呢?下面…...

关键领域软件测试的突围之路:如何破解安全与效率的平衡难题
在数字化浪潮席卷全球的今天,软件系统已成为国家关键领域的核心战斗力。不同于普通商业软件,这些承载着国家安全使命的软件系统面临着前所未有的质量挑战——如何在确保绝对安全的前提下,实现高效测试与快速迭代?这一命题正考验着…...

云原生玩法三问:构建自定义开发环境
云原生玩法三问:构建自定义开发环境 引言 临时运维一个古董项目,无文档,无环境,无交接人,俗称三无。 运行设备的环境老,本地环境版本高,ssh不过去。正好最近对 腾讯出品的云原生 cnb 感兴趣&…...
【JavaSE】多线程基础学习笔记
多线程基础 -线程相关概念 程序(Program) 是为完成特定任务、用某种语言编写的一组指令的集合简单的说:就是我们写的代码 进程 进程是指运行中的程序,比如我们使用QQ,就启动了一个进程,操作系统就会为该进程分配内存…...

Qemu arm操作系统开发环境
使用qemu虚拟arm硬件比较合适。 步骤如下: 安装qemu apt install qemu-system安装aarch64-none-elf-gcc 需要手动下载,下载地址:https://developer.arm.com/-/media/Files/downloads/gnu/13.2.rel1/binrel/arm-gnu-toolchain-13.2.rel1-x…...
【Elasticsearch】Elasticsearch 在大数据生态圈的地位 实践经验
Elasticsearch 在大数据生态圈的地位 & 实践经验 1.Elasticsearch 的优势1.1 Elasticsearch 解决的核心问题1.1.1 传统方案的短板1.1.2 Elasticsearch 的解决方案 1.2 与大数据组件的对比优势1.3 关键优势技术支撑1.4 Elasticsearch 的竞品1.4.1 全文搜索领域1.4.2 日志分析…...
面试高频问题
文章目录 🚀 消息队列核心技术揭秘:从入门到秒杀面试官1️⃣ Kafka为何能"吞云吐雾"?性能背后的秘密1.1 顺序写入与零拷贝:性能的双引擎1.2 分区并行:数据的"八车道高速公路"1.3 页缓存与批量处理…...
Java详解LeetCode 热题 100(26):LeetCode 142. 环形链表 II(Linked List Cycle II)详解
文章目录 1. 题目描述1.1 链表节点定义 2. 理解题目2.1 问题可视化2.2 核心挑战 3. 解法一:HashSet 标记访问法3.1 算法思路3.2 Java代码实现3.3 详细执行过程演示3.4 执行结果示例3.5 复杂度分析3.6 优缺点分析 4. 解法二:Floyd 快慢指针法(…...