我的编程空间,编程开发者的网络收藏夹
学习永远不晚

MyBatis插件的原理是什么

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

MyBatis插件的原理是什么

本篇文章为大家展示了MyBatis插件的原理是什么,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。

 插件原理分析

mybatis插件涉及到的几个类:

MyBatis插件的原理是什么

我将以 Executor 为例,分析 MyBatis 是如何为 Executor 实例植入插件的。Executor 实例是在开启 SqlSession  时被创建的,因此,我们从源头进行分析。先来看一下 SqlSession 开启的过程。

public SqlSession openSession() {     return openSessionFromDataSource(configuration.getDefaultExecutorType(), null, false); }  private SqlSession openSessionFromDataSource(ExecutorType execType, TransactionIsolationLevel level, boolean autoCommit) {     Transaction tx = null;     try {         // 省略部分逻辑                  // 创建 Executor         final Executor executor = configuration.newExecutor(tx, execType);         return new DefaultSqlSession(configuration, executor, autoCommit);     }      catch (Exception e) {...}      finally {...} }

Executor 的创建过程封装在 Configuration 中,我们跟进去看看看。

// Configuration类中 public Executor newExecutor(Transaction transaction, ExecutorType executorType) {     executorType = executorType == null ? defaultExecutorType : executorType;     executorType = executorType == null ? ExecutorType.SIMPLE : executorType;     Executor executor;          // 根据 executorType 创建相应的 Executor 实例     if (ExecutorType.BATCH == executorType) {...}      else if (ExecutorType.REUSE == executorType) {...}      else {         executor = new SimpleExecutor(this, transaction);     }     if (cacheEnabled) {         executor = new CachingExecutor(executor);     }          // 植入插件     executor = (Executor) interceptorChain.pluginAll(executor);     return executor; }

如上,newExecutor 方法在创建好 Executor 实例后,紧接着通过拦截器链 interceptorChain 为 Executor  实例植入代理逻辑。那下面我们看一下 InterceptorChain 的代码是怎样的。

public class InterceptorChain {     private final List<Interceptor> interceptors = new ArrayList<Interceptor>();     public Object pluginAll(Object target) {         // 遍历拦截器集合         for (Interceptor interceptor : interceptors) {             // 调用拦截器的 plugin 方法植入相应的插件逻辑             target = interceptor.plugin(target);         }         return target;     }          public void addInterceptor(Interceptor interceptor) {         interceptors.add(interceptor);     }          public List<Interceptor> getInterceptors() {         return Collections.unmodifiableList(interceptors);     } }

上面的for循环代表了只要是插件,都会以责任链的方式逐一执行(别指望它能跳过某个节点),所谓插件,其实就类似于拦截器。

这里就用到了责任链设计模式,责任链设计模式就相当于我们在OA系统里发起审批,领导们一层一层进行审批。

以上是 InterceptorChain 的全部代码,比较简单。它的 pluginAll 方法会调用具体插件的 plugin  方法植入相应的插件逻辑。如果有多个插件,则会多次调用 plugin 方法,最终生成一个层层嵌套的代理类。形如下面:

MyBatis插件的原理是什么

当 Executor 的某个方法被调用的时候,插件逻辑会先行执行。执行顺序由外而内,比如上图的执行顺序为 plugin3 &rarr; plugin2 &rarr;  Plugin1 &rarr; Executor。

plugin 方法是由具体的插件类实现,不过该方法代码一般比较固定,所以下面找个示例分析一下。

// TianPlugin类 public Object plugin(Object target) {     return Plugin.wrap(target, this); }  //Plugin public static Object wrap(Object target, Interceptor interceptor) {          Map<Class<?>, Set<Method>> signatureMap = getSignatureMap(interceptor);     Class<?> type = target.getClass();     // 获取目标类实现的接口     Class<?>[] interfaces = getAllInterfaces(type, signatureMap);     if (interfaces.length > 0) {         // 通过 JDK 动态代理为目标类生成代理类         return Proxy.newProxyInstance(             type.getClassLoader(),             interfaces,             new Plugin(target, interceptor, signatureMap));     }     return target; }

如上,plugin 方法在内部调用了 Plugin 类的 wrap 方法,用于为目标对象生成代理。Plugin 类实现了  InvocationHandler 接口,因此它可以作为参数传给 Proxy 的 newProxyInstance 方法。

到这里,关于插件植入的逻辑就分析完了。接下来,我们来看看插件逻辑是怎样执行的。

执行插件逻辑

Plugin 实现了 InvocationHandler 接口,因此它的 invoke 方法会拦截所有的方法调用。invoke  方法会对所拦截的方法进行检测,以决定是否执行插件逻辑。该方法的逻辑如下:

//在Plugin类中 public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {     try {                  Set<Method> methods = signatureMap.get(method.getDeclaringClass());         // 检测方法列表是否包含被拦截的方法         if (methods != null && methods.contains(method)) {             // 执行插件逻辑             return interceptor.intercept(new Invocation(target, method, args));         }         // 执行被拦截的方法         return method.invoke(target, args);     } catch (Exception e) {         throw ExceptionUtil.unwrapThrowable(e);     } }

invoke 方法的代码比较少,逻辑不难理解。首先,invoke 方法会检测被拦截方法是否配置在插件的 @Signature  注解中,若是,则执行插件逻辑,否则执行被拦截方法。插件逻辑封装在 intercept 中,该方法的参数类型为 Invocation。Invocation  主要用于存储目标类,方法以及方法参数列表。下面简单看一下该类的定义。

public class Invocation {      private final Object target;     private final Method method;     private final Object[] args;      public Invocation(Object target, Method method, Object[] args) {         this.target = target;         this.method = method;         this.args = args;     }     // 省略部分代码     public Object proceed() throws InvocationTargetException, IllegalAccessException {         //反射调用被拦截的方法         return method.invoke(target, args);     } }

关于插件的执行逻辑就分析到这,整个过程不难理解,大家简单看看即可。

自定义插件

下面为了让大家更好的理解Mybatis的插件机制,我们来模拟一个慢sql监控的插件。

 @Intercepts({@Signature(type = StatementHandler.class, method = "prepare", args = {Connection.class, Integer.class})}) public class SlowSqlPlugin implements Interceptor {      private long slowTime;      //拦截后需要处理的业务     @Override     public Object intercept(Invocation invocation) throws Throwable {         //通过StatementHandler获取执行的sql         StatementHandler statementHandler = (StatementHandler) invocation.getTarget();         BoundSql boundSql = statementHandler.getBoundSql();         String sql = boundSql.getSql();          long start = System.currentTimeMillis();         //结束拦截         Object proceed = invocation.proceed();         long end = System.currentTimeMillis();         long f = end - start;         System.out.println(sql);         System.out.println("耗时=" + f);         if (f > slowTime) {             System.out.println("本次数据库操作是慢查询,sql是:");             System.out.println(sql);         }         return proceed;     }      //获取到拦截的对象,底层也是通过代理实现的,实际上是拿到一个目标代理对象     @Override     public Object plugin(Object target) {         //触发intercept方法         return Plugin.wrap(target, this);     }      //设置属性     @Override     public void setProperties(Properties properties) {         //获取我们定义的慢sql的时间阈值slowTime         this.slowTime = Long.parseLong(properties.getProperty("slowTime"));     } }

然后把这个插件类注入到容器中。

MyBatis插件的原理是什么

然后我们来执行查询的方法。

MyBatis插件的原理是什么

耗时28秒的,大于我们定义的10毫秒,那这条SQL就是我们认为的慢SQL。

通过这个插件,我们就能很轻松的理解setProperties()方法是做什么的了。

回顾分页插件

也是实现mybatis接口Interceptor。

@SuppressWarnings({"rawtypes", "unchecked"}) @Intercepts(     {         @Signature(type = Executor.class, method = "query", args = {MappedStatement.class, Object.class, RowBounds.class, ResultHandler.class}),         @Signature(type = Executor.class, method = "query", args = {MappedStatement.class, Object.class, RowBounds.class, ResultHandler.class, CacheKey.class, BoundSql.class}),     } ) public class PageInterceptor implements Interceptor {         @Override     public Object intercept(Invocation invocation) throws Throwable {         ...     }

intercept方法中

MyBatis插件的原理是什么

//AbstractHelperDialect类中 @Override public String getPageSql(MappedStatement ms, BoundSql boundSql, Object parameterObject, RowBounds rowBounds, CacheKey pageKey) {         String sql = boundSql.getSql();         Page page = getLocalPage();         //支持 order by         String orderBy = page.getOrderBy();         if (StringUtil.isNotEmpty(orderBy)) {             pageKey.update(orderBy);             sql = OrderByParser.converToOrderBySql(sql, orderBy);         }         if (page.isOrderByOnly()) {             return sql;         }         //获取分页sql         return getPageSql(sql, page, pageKey);  } //模板方法模式中的钩子方法  public abstract String getPageSql(String sql, Page page, CacheKey pageKey);

AbstractHelperDialect类的实现类有如下(也就是此分页插件支持的数据库就以下几种):

MyBatis插件的原理是什么

我们用的是MySQL。这里也有与之对应的。

@Override    public String getPageSql(String sql, Page page, CacheKey pageKey) {        StringBuilder sqlBuilder = new StringBuilder(sql.length() + 14);        sqlBuilder.append(sql);        if (page.getStartRow() == 0) {            sqlBuilder.append(" LIMIT ? ");        } else {            sqlBuilder.append(" LIMIT ?, ? ");        }        pageKey.update(page.getPageSize());        return sqlBuilder.toString();    }

到这里我们就知道了,它无非就是在我们执行的SQL上再拼接了Limit罢了。同理,Oracle也就是使用rownum来处理分页了。下面是Oracle处理分页

@Override     public String getPageSql(String sql, Page page, CacheKey pageKey) {         StringBuilder sqlBuilder = new StringBuilder(sql.length() + 120);         if (page.getStartRow() > 0) {             sqlBuilder.append("SELECT * FROM ( ");         }         if (page.getEndRow() > 0) {             sqlBuilder.append(" SELECT TMP_PAGE.*, ROWNUM ROW_ID FROM ( ");         }         sqlBuilder.append(sql);         if (page.getEndRow() > 0) {             sqlBuilder.append(" ) TMP_PAGE WHERE ROWNUM <= ? ");         }         if (page.getStartRow() > 0) {             sqlBuilder.append(" ) WHERE ROW_ID > ? ");         }         return sqlBuilder.toString();     }

其他数据库分页操作类似。关于具体原理分析,这里就没必要赘述了,因为分页插件源代码里注释基本上全是中文。

Mybatis插件应用场景

  • 水平分表

  • 权限控制

  • 数据的加解密

总结

Spring-Boot+Mybatis继承了分页插件,以及使用案例、插件的原理分析、源码分析、如何自定义插件。

涉及到技术点:JDK动态代理、责任链设计模式、模板方法模式。

Mybatis插件关键对象总结:

  • Inteceptor接口:自定义拦截必须实现的类。

  • InterceptorChain:存放插件的容器。

  • Plugin:h对象,提供创建代理类的方法。

  • Invocation:对被代理对象的封装。

上述内容就是MyBatis插件的原理是什么,你们学到知识或技能了吗?如果还想学到更多技能或者丰富自己的知识储备,欢迎关注编程网行业资讯频道。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

MyBatis插件的原理是什么

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

PHP插件机制的原理是什么

这篇文章主要讲解了“PHP插件机制的原理是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“PHP插件机制的原理是什么”吧!插件,亦即Plug-in,是指一类特定的功能模块(通常由第三方开发
2023-06-17

Vue3.0插件执行原理是什么

这篇文章主要介绍“Vue3.0插件执行原理是什么”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Vue3.0插件执行原理是什么”文章能帮助大家解决问题。一、编写插件Vue项目能够使用很多插件来丰富自己
2023-06-29

mybatis-plugin插件执行原理解析

这篇文章主要介绍了mybatis-plugin插件执行原理,我们就需要来研究下Executor,ParameterHandler,ResultSetHandler,StatementHandler这4个对象的具体跟sql相关的方法,然后再进行修改,就可以直接起到aop的作用,需要的朋友可以参考下
2022-11-13

MyBatis分页插件PageHelper的使用与原理

提到插件相信大家都知道,插件的存在主要是用来改变或者增强原有的功能,MyBatis中也一样,下面这篇文章主要给大家介绍了关于Mybatis第三方PageHelper分页插件的使用与原理,需要的朋友可以参考下
2023-02-24

mybatis流查询的原理是什么

MyBatis是一个基于Java的持久层框架,其流查询的原理是利用数据库的游标功能来一次性获取大量数据,减少内存的消耗和提高查询效率。在MyBatis中,使用流查询可以通过设置statement.fetchSize属性来实现。该属性指定了
mybatis流查询的原理是什么
2024-03-12

mybatis二级缓存的原理是什么

MyBatis的二级缓存是指缓存在SqlSessionFactory级别的缓存,可以被多个SqlSession共享。其原理如下:1. 当一个查询语句被执行时,MyBatis会首先检查二级缓存中是否存在相应的缓存数据,如果存在,则直接返回缓存
2023-10-09

mybatis自动封装的原理是什么

MyBatis的自动封装是通过使用Java的反射机制来实现的。MyBatis在执行查询操作时,会通过Java的反射机制,根据查询结果集的列名和Java对象的属性名进行匹配,将查询结果自动封装到相应的Java对象中。具体的步骤如下:MyB
2023-10-27

Vue插槽的实现原理是什么

这篇文章主要介绍“Vue插槽的实现原理是什么”,在日常操作中,相信很多人在Vue插槽的实现原理是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Vue插槽的实现原理是什么”的疑惑有所帮助!接下来,请跟着小编
2023-06-20

Eclipse的插件MyBatis Editor有什么用

这篇文章将为大家详细讲解有关Eclipse的插件MyBatis Editor有什么用,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。MyBatis Editor是一个Eclipse的插件,用来编辑MyBat
2023-06-17

Mybatis SqlSessionFactory初始化原理是什么

这篇文章主要讲解了“Mybatis SqlSessionFactory初始化原理是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Mybatis SqlSessionFactory初始化原
2023-06-20

php中插入排序的原理是什么

php中插入排序的原理是什么?很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。php有什么用php是一个嵌套的缩写名称,是英文超级文本预处理语言,它的语法混合了C、
2023-06-14

Vue3插槽Slot的实现原理是什么

这篇文章主要介绍了Vue3插槽Slot的实现原理是什么的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Vue3插槽Slot的实现原理是什么文章都会有所收获,下面我们一起来看看吧。Vue官方对插槽的定义Vue 实现
2023-07-02

mybatis工作原理及流程是什么

MyBatis是一种持久层框架,它的工作原理及流程如下:1. 配置文件加载:MyBatis首先读取配置文件(通常是mybatis-config.xml),该文件包含了数据库连接信息、映射文件路径等配置。2. 映射文件加载:MyBatis根据
2023-08-14

Mybatis防止sql注入原理是什么

这篇文章主要讲解了“Mybatis防止sql注入原理是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Mybatis防止sql注入原理是什么”吧!Mybatis防止sql注入原理SQL 注
2023-06-22

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录