您现在的位置是:亿华云 > 系统运维
建议收藏,MyBatis插件原理详解
亿华云2025-10-03 06:35:20【系统运维】5人已围观
简介插件原理分析mybatis插件涉及到的几个类:我将以 Executor 为例,分析 MyBatis 是如何为 Executor 实例植入插件的。Executor 实例是在开启 SqlSession 时
插件原理分析
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; } /** 添加插件实例到 interceptors 集合中 */ public void addInterceptor(Interceptor interceptor) { interceptors.add(interceptor); } /** 获取插件列表 */ public List<Interceptor> getInterceptors() { return Collections.unmodifiableList(interceptors); } }上面的for循环代表了只要是插件,都会以责任链的原解方式逐一执行(别指望它能跳过某个节点),所谓插件,理详其实就类似于拦截器。
这里就用到了责任链设计模式,责任链设计模式就相当于我们在OA系统里发起审批,源码下载领导们一层一层进行审批。
以上是 InterceptorChain 的全部代码,比较简单。它的 pluginAll 方法会调用具体插件的 plugin 方法植入相应的插件逻辑。如果有多个插件,则会多次调用 plugin 方法,最终生成一个层层嵌套的代理类。形如下面:
当 Executor 的某个方法被调用的时候,插件逻辑会先行执行。执行顺序由外而内,比如上图的执行顺序为 plugin3 → plugin2 → Plugin1 → Executor。
plugin 方法是由具体的插件类实现,不过该方法代码一般比较固定,所以下面找个示例分析一下。
// TianPlugin类 public Object plugin(Object target) { return Plugin.wrap(target, this); } //Plugin public static Object wrap(Object target, Interceptor interceptor) { /* * 获取插件类 @Signature 注解内容,并生成相应的映射结构。形如下面: * { * Executor.class : [query, update, commit], * ParameterHandler.class : [getParameterObject, setParameters] * } */ 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 { /* * 获取被拦截方法列表,比如: * signatureMap.get(Executor.class),可能返回 [query, update, commit] */ 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监控的插件。
/** * 慢查询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")); } }然后把这个插件类注入到容器中。
然后我们来执行查询的方法。
耗时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方法中
AbstractHelperDialect类的实现类有如下(也就是此分页插件支持的数据库就以下几种):
我们用的是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:对被代理对象的封装。本文转载自微信公众号「Java后端技术全栈」,可以通过以下二维码关注。转载本文请联系Java后端技术全栈公众号。
很赞哦!(48446)
上一篇: 数据中心结构化布线指南
下一篇: 提高数据中心能源可持续性:短期与长期策略