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)、定时器/计…...
智慧隧道和智慧交通
通过引入先进的物联网技术,将各种硬件设备如传感器、摄像头、控制系统等有效地连接并管理起来,以实现道路安全和交通流畅的目标。这些设备将能够实时监控和控制隧道内的各种设备和系统,从而提高道路安全、提升驾驶体验并降低管理成本。 在这个…...
利用ngx_stream_return_module构建简易 TCP/UDP 响应网关
一、模块概述 ngx_stream_return_module 提供了一个极简的指令: return <value>;在收到客户端连接后,立即将 <value> 写回并关闭连接。<value> 支持内嵌文本和内置变量(如 $time_iso8601、$remote_addr 等)&a…...

Appium+python自动化(十六)- ADB命令
简介 Android 调试桥(adb)是多种用途的工具,该工具可以帮助你你管理设备或模拟器 的状态。 adb ( Android Debug Bridge)是一个通用命令行工具,其允许您与模拟器实例或连接的 Android 设备进行通信。它可为各种设备操作提供便利,如安装和调试…...
Spring Boot 实现流式响应(兼容 2.7.x)
在实际开发中,我们可能会遇到一些流式数据处理的场景,比如接收来自上游接口的 Server-Sent Events(SSE) 或 流式 JSON 内容,并将其原样中转给前端页面或客户端。这种情况下,传统的 RestTemplate 缓存机制会…...
从零实现富文本编辑器#5-编辑器选区模型的状态结构表达
先前我们总结了浏览器选区模型的交互策略,并且实现了基本的选区操作,还调研了自绘选区的实现。那么相对的,我们还需要设计编辑器的选区表达,也可以称为模型选区。编辑器中应用变更时的操作范围,就是以模型选区为基准来…...

UE5 学习系列(三)创建和移动物体
这篇博客是该系列的第三篇,是在之前两篇博客的基础上展开,主要介绍如何在操作界面中创建和拖动物体,这篇博客跟随的视频链接如下: B 站视频:s03-创建和移动物体 如果你不打算开之前的博客并且对UE5 比较熟的话按照以…...
1688商品列表API与其他数据源的对接思路
将1688商品列表API与其他数据源对接时,需结合业务场景设计数据流转链路,重点关注数据格式兼容性、接口调用频率控制及数据一致性维护。以下是具体对接思路及关键技术点: 一、核心对接场景与目标 商品数据同步 场景:将1688商品信息…...
React Native在HarmonyOS 5.0阅读类应用开发中的实践
一、技术选型背景 随着HarmonyOS 5.0对Web兼容层的增强,React Native作为跨平台框架可通过重新编译ArkTS组件实现85%以上的代码复用率。阅读类应用具有UI复杂度低、数据流清晰的特点。 二、核心实现方案 1. 环境配置 (1)使用React Native…...

学校招生小程序源码介绍
基于ThinkPHPFastAdminUniApp开发的学校招生小程序源码,专为学校招生场景量身打造,功能实用且操作便捷。 从技术架构来看,ThinkPHP提供稳定可靠的后台服务,FastAdmin加速开发流程,UniApp则保障小程序在多端有良好的兼…...
vue3 字体颜色设置的多种方式
在Vue 3中设置字体颜色可以通过多种方式实现,这取决于你是想在组件内部直接设置,还是在CSS/SCSS/LESS等样式文件中定义。以下是几种常见的方法: 1. 内联样式 你可以直接在模板中使用style绑定来设置字体颜色。 <template><div :s…...
将对透视变换后的图像使用Otsu进行阈值化,来分离黑色和白色像素。这句话中的Otsu是什么意思?
Otsu 是一种自动阈值化方法,用于将图像分割为前景和背景。它通过最小化图像的类内方差或等价地最大化类间方差来选择最佳阈值。这种方法特别适用于图像的二值化处理,能够自动确定一个阈值,将图像中的像素分为黑色和白色两类。 Otsu 方法的原…...