博客
关于我
MyBatis拦截器原理探究
阅读量:670 次
发布时间:2019-03-16

本文共 3618 字,大约阅读时间需要 12 分钟。

MyBatis 拦截器介绍

MyBatis 提供了一个强大的插件系统,允许开发者在特定的执行点进行拦截调用。虽然这个功能被称为插件,但它实际上是一个非常强大的拦截器功能。拦截器可以拦截 MyBatis 中的多个关键操作,包括但不限于以下内容:

1. 拦截的目标

拦截器拦截以下关键接口中的方法:

  • Executor 接口:包括 updatequerycommitrollback 等方法。这些方法通常用于执行数据库操作。
  • ParameterHandler 接口:拦截数据参数的处理,主要涉及 getParameterObjectsetParameters 方法。
  • ResultSetHandler 接口:拦截结果集的处理,主要涉及 handleResultSetshandleOutputParameters 方法。
  • StatementHandler 接口:拦截 SQL 语法构建的相关操作,包括 prepareparameterizebatchupdatequery 方法。

这些拦截器可以用于各种场景,例如数据带入、结果集处理、事务管理等。


2. 拦截器的使用

2.1 拦截器接口定义

MyBatis 的拦截器接口定义为:

public interface Interceptor {    Object intercept(Invocation invocation) throws Throwable;    Object plugin(Object target);    void setProperties(Properties properties);}

拦截器接口只有三个方法:

  • intercept(Invocation invocation):拦截具体的方法调用。
  • plugin(Object target):对目标进行适配处理。
  • setProperties(Properties properties):设置拦截器的属性。
2.2 拦截器配置

全局配置中,可以通过 XML 或注解的方式注册拦截器。例如,以下是 MyBatis 的一个拦截器示例:

@Intercepts({@Signature(type = Executor.class, method = "update", args = {MappedStatement.class, Object.class})}public class ExamplePlugin implements Interceptor {    public Object intercept(Invocation invocation) throws Throwable {        return invocation.proceed();    }    public Object plugin(Object target) {        return Plugin.wrap(target, this);    }    public void setProperties(Properties properties) {}}

在 XML 配置中,拦截器可以通过如下方式添加:

2.3 拦截器的实现

拦截器的核心逻辑通常在 plugin 方法中实现。例如,在上面的示例中,plugin 方法使用了 Plugin.wrap(target, this) 来创建一个动态代理,这样可以在拦截器激活时对目标进行适配。


3. 拦截器的内部工作原理

3.1 配置解析

MyBatis 在解析 XML 配置时,会调用 pluginElement 方法处理拦截器配置。解析逻辑如下:

private void pluginElement(XNode parent) throws Exception {    for (XNode child : parent.getChildren()) {        String interceptor = child.getStringAttribute("interceptor");        Properties properties = child.getChildrenAsProperties();        Interceptor interceptorInstance = (Interceptor) resolveClass(interceptor).newInstance();        interceptorInstance.setProperties(properties);        configuration.addInterceptor(interceptorInstance);    }}
3.2 拦截器链

拦截器链位于 Configuration 类中,负责协调多个拦截器的运行。拦截器链的主要功能包括:

  • 遍历所有注册的拦截器。
  • 调用每个拦截器的 plugin 方法对目标进行处理。
3.3 方法创建与拦截

MyBatis 在创建 ExecutorParameterHandlerResultSetHandlerStatementHandler 时,会调用拦截器链的 pluginAll 方法:

  • parameterHandler = newParameterHandler(...)
  • resultSetHandler = newResultSetHandler(...)
  • statementHandler = newStatementHandler(...)
  • executor = newExecutor(...)

这些方法在执行时都会传递给拦截器链,进行插件处理。


4. 通过代码实现拦截

MyBatis 拦截器的实现可以通过反射和动态代理实现。Plugin.wrap(target, interceptor) 方法会生成一个动态代理,代理委托拦截器的 intercept 方法进行处理。

对于 StatementHandler 的拦截,可以实现如下逻辑:

public Object plugin(Object target) {    return Plugin.wrap(target, new Plugin() {        public Object intercept(Invocation invocation) throws Throwable {            // 自定义 SQL 语句处理逻辑            return invocation.proceed();        }    });}

源码解析

4.1 插件代理

Plugin.wrap 方法通过动态代理技术创建一个代理实例,核心逻辑包括:

  • 获取拦截器的签名信息(通过 getSignatureMap 方法)。
  • 根据目标实例的类型与签名信息生成代理。
  • 创建一个动态代理,代理调用目标实例的方法。
  • 4.2 签名信息处理

    getSignatureMap 方法用于解析拦截器的 @Intercepts@Signature 注解,返回一个映射,记录每个接口和方法的签名。

    4.3 动态代理

    动态代理的 invoke 方法实现了拦截逻辑:

    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {    try {        Set
    methods = signatureMap.get(method.getDeclaringClass()); if (methods != null && methods.contains(method)) { return interceptor.intercept(new Invocation(target, method, args)); } else { return method.invoke(target, args); } } catch (Exception e) { throw ExceptionUtil.unwrapThrowable(e); }}

    总结

    MyBatis 拦截器是一个非常强大的功能,可以用于扩展和定制 MyBatis 的行为。通过实现拦截器接口,开发者可以拦截关键操作并添加自定义逻辑。拦截器的核心优势在于其灵活性和可扩展性,适用于从数据处理、结果集处理到事务管理等多个场景。

    转载地址:http://cpiqz.baihongyu.com/

    你可能感兴趣的文章
    NIFI从MySql中离线读取数据再导入到MySql中_不带分页处理_01_QueryDatabaseTable获取数据_原0036---大数据之Nifi工作笔记0064
    查看>>
    NIFI从MySql中离线读取数据再导入到MySql中_无分页功能_02_转换数据_分割数据_提取JSON数据_替换拼接SQL_添加分页---大数据之Nifi工作笔记0037
    查看>>
    NIFI从Oracle11G同步数据到Mysql_亲测可用_解决数据重复_数据跟源表不一致的问题---大数据之Nifi工作笔记0065
    查看>>
    NIFI从PostGresql中离线读取数据再导入到MySql中_带有数据分页获取功能_不带分页不能用_NIFI资料太少了---大数据之Nifi工作笔记0039
    查看>>
    nifi使用过程-常见问题-以及入门总结---大数据之Nifi工作笔记0012
    查看>>
    NIFI分页获取Mysql数据_导入到Hbase中_并可通过phoenix客户端查询_含金量很高的一篇_搞了好久_实际操作05---大数据之Nifi工作笔记0045
    查看>>
    NIFI分页获取Postgresql数据到Hbase中_实际操作---大数据之Nifi工作笔记0049
    查看>>
    NIFI同步MySql数据_到SqlServer_错误_驱动程序无法通过使用安全套接字层(SSL)加密与SQL Server_Navicat连接SqlServer---大数据之Nifi工作笔记0047
    查看>>
    NIFI同步MySql数据源数据_到原始库hbase_同时对数据进行实时分析处理_同步到清洗库_实际操作06---大数据之Nifi工作笔记0046
    查看>>
    Nifi同步过程中报错create_time字段找不到_实际目标表和源表中没有这个字段---大数据之Nifi工作笔记0066
    查看>>
    NIFI大数据进阶_FlowFile拓扑_对FlowFile内容和属性的修改删除添加_介绍和描述_以及实际操作---大数据之Nifi工作笔记0023
    查看>>
    NIFI大数据进阶_FlowFile生成器_GenerateFlowFile处理器_ReplaceText处理器_处理器介绍_处理过程说明---大数据之Nifi工作笔记0019
    查看>>
    NIFI大数据进阶_FlowFile生成器_GenerateFlowFile处理器_ReplaceText处理器_实际操作---大数据之Nifi工作笔记0020
    查看>>
    NIFI大数据进阶_Json内容转换为Hive支持的文本格式_实际操作_02---大数据之Nifi工作笔记0032
    查看>>
    NIFI大数据进阶_Json内容转换为Hive支持的文本格式_操作方法说明_01_EvaluteJsonPath处理器---大数据之Nifi工作笔记0031
    查看>>
    NIFI大数据进阶_Kafka使用相关说明_实际操作Kafka消费者处理器_来消费kafka数据---大数据之Nifi工作笔记0037
    查看>>
    NIFI大数据进阶_Kafka使用相关说明_实际操作Kafka生产者---大数据之Nifi工作笔记0036
    查看>>
    NIFI大数据进阶_NIFI的模板和组的使用-介绍和实际操作_创建组_嵌套组_模板创建下载_导入---大数据之Nifi工作笔记0022
    查看>>
    NIFI大数据进阶_NIFI监控功能实际操作_Summary查看系统和处理器运行情况_viewDataProvenance查看_---大数据之Nifi工作笔记0026
    查看>>
    NIFI大数据进阶_NIFI监控的强大功能介绍_处理器面板_进程组面板_summary监控_data_provenance事件源---大数据之Nifi工作笔记0025
    查看>>