mybatis拦截器(插件)原理

所谓的插件其实就是方法拦截器,mybatis四大组件(Executor,StatementHandler,ParameterHandler,ResultSetHandler)在运行过程中,对四大组件方法拦截并增强,底层用的jdk的动态代理实现,常见的插件有分页插件,mybatis-plus中的MybatisPlusInterceptor,自定义插件,如在插入数据库前设置一些默认的字段(这个在mybatis-plus中不需要拦截器实现)等。

普通方法的拦截

先用mybatis中的Interceptor对普通方法拦截,加强理解

接口:

package mybatis;

/**
 * 接口,jdk动态代理必须有接口
 */
public interface TestInterface {
    String test();
    String test(String arg1);
}

实现类:

package mybatis;

/**
 * 实现类
 * 我们这里拟定目标,只增强待参数的test 方法
 */
public class TestInterfaceImpl implements TestInterface{

    @Override
    public String test() {
        return "无参";
    }

    @Override
    public String test(String arg1) {
        return arg1;
    }
}

方法拦截器

package mybatis;

import org.apache.ibatis.plugin.Interceptor;
import org.apache.ibatis.plugin.Intercepts;
import org.apache.ibatis.plugin.Invocation;
import org.apache.ibatis.plugin.Signature;

import java.util.Properties;

/**
 * 拦截器,作用:
 * 指定拦截哪个类的哪个方法(Intercepts注解)
 * 方法的增强逻辑(intercept方法)
 * 对目标对象包装,创建动态代理代替目标对象(plugin方法)
 */
// 指定拦截哪个接口的哪个方法
@Intercepts(
        {
                @Signature(type = TestInterface.class, method = "test", args = {String.class})
        }
)
public class MyInterceptor implements Interceptor {

        // 增强逻辑写在这里,invocation 里可以获取目标对象,参数信息
        @Override
        public Object intercept(Invocation invocation) throws Throwable {
                Object[] args = invocation.getArgs();
                // 拦截的是TestInterface接口实现类的方法,所以可以直接强转
                TestInterface target = (TestInterface) invocation.getTarget();
                System.out.println("进入方法拦截。。。。。");
                return target.test(args[0].toString());
        }
        // 创建代理对象提供给客户端,底层用的jdk的动态代理
        @Override
        public Object plugin(Object target) {
                return Interceptor.super.plugin(target);
        }
        // 目前没用上
        @Override
        public void setProperties(Properties properties) {
                Interceptor.super.setProperties(properties);
        }
}

单元测试类:

@Test
    public void testPlug() {
        // 需要增强的对象
        TestInterface testInterface = new TestInterfaceImpl();
        // 创建两个拦截器对象,一个目标可以被多个拦截器包装,一层一层的执行
        Interceptor myInterceptor = new MyInterceptor();
        Interceptor myInterceptor2 = new MyInterceptor();
        //执行一次wrap方法就创建一个代理对象,wrap执行后相当于:Proxy1(target)
        testInterface = (TestInterface) Plugin.wrap(testInterface, myInterceptor);
        // 第二次wrap方法执行后相当于:Proxy(Proxy1(target))
        testInterface = (TestInterface) Plugin.wrap(testInterface, myInterceptor2);
        // 执行被拦截的方法
        System.out.println(testInterface.test("hello"));
    }

结果:

进入方法拦截。。。。。
进入方法拦截。。。。。
hello

Process finished with exit code 0

可以看到两个增强器的方法都执行了

Mybatis 插件原理分析

mybatis 框架只对四大组件设置了插件机制
1、组件是怎么添加到框架中的,拦截器对象保存在哪?
项目启动时解析mybatis 配置文件,解析xml中的plugins 标签或者配置类中通过java代码直接添加。
保存在一个集合中,InterceptorChain 类中的List interceptors 成员变量中

2、何时创建代理对目标进行包装?
Configuration 创建四大组件时,如果有插件,就会创建代理对象,可以是多个,多个代理对象嵌套形成责任链,源码如下:

public ParameterHandler newParameterHandler(MappedStatement mappedStatement, Object parameterObject, BoundSql boundSql) {
    ParameterHandler parameterHandler = mappedStatement.getLang().createParameterHandler(mappedStatement, parameterObject, boundSql);
    // 用所有的拦截器包装
    parameterHandler = (ParameterHandler) interceptorChain.pluginAll(parameterHandler);
    return parameterHandler;
  }

  public ResultSetHandler newResultSetHandler(Executor executor, MappedStatement mappedStatement, RowBounds rowBounds, ParameterHandler parameterHandler,
      ResultHandler resultHandler, BoundSql boundSql) {
    ResultSetHandler resultSetHandler = new DefaultResultSetHandler(executor, mappedStatement, parameterHandler, resultHandler, boundSql, rowBounds);
        // 用所有的拦截器包装
    resultSetHandler = (ResultSetHandler) interceptorChain.pluginAll(resultSetHandler);
    return resultSetHandler;
  }

  public StatementHandler newStatementHandler(Executor executor, MappedStatement mappedStatement, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler, BoundSql boundSql) {
    StatementHandler statementHandler = new RoutingStatementHandler(executor, mappedStatement, parameterObject, rowBounds, resultHandler, boundSql);
        // 用所有的拦截器包装
    statementHandler = (StatementHandler) interceptorChain.pluginAll(statementHandler);
    return statementHandler;
  }

  public Executor newExecutor(Transaction transaction) {
    return newExecutor(transaction, defaultExecutorType);
  }

  public Executor newExecutor(Transaction transaction, ExecutorType executorType) {
    executorType = executorType == null ? defaultExecutorType : executorType;
    executorType = executorType == null ? ExecutorType.SIMPLE : executorType;
    Executor executor;
    if (ExecutorType.BATCH == executorType) {
      executor = new BatchExecutor(this, transaction);
    } else if (ExecutorType.REUSE == executorType) {
      executor = new ReuseExecutor(this, transaction);
    } else {
      executor = new SimpleExecutor(this, transaction);
    }
    if (cacheEnabled) {
      executor = new CachingExecutor(executor);
    }
        // 用所有的拦截器包装
    executor = (Executor) interceptorChain.pluginAll(executor);
    return executor;
  }

包装源码:
interceptorChain.pluginAll 最终会调用到Plugin.wrap(target, this) 方法,过程比较简单, wrap() 方法源码

package org.apache.ibatis.plugin;
/**
 * @author Clinton Begin
 */
public class Plugin implements InvocationHandler {

  private final Object target;
  private final Interceptor interceptor;
  private final Map<Class<?>, Set<Method>> signatureMap;
    // 创建动态代理,将interceptor(拦截器中对目标增强逻辑) 包装成Plugin对象 (动态代理的InvocationHandler)
  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) {
      return Proxy.newProxyInstance(
          type.getClassLoader(),
          interfaces,
          new Plugin(target, interceptor, signatureMap));
    }
    return target;
  }
// jdk 动态代理,代理对象执行目标方法前先执行invoke方法
  @Override
  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)) {
          //  @Signature 注解中标注了的方法才回去拦截
        return interceptor.intercept(new Invocation(target, method, args));
      }
      // 否则直接执行目标方法
      return method.invoke(target, args);
    } catch (Exception e) {
      throw ExceptionUtil.unwrapThrowable(e);
    }
  }
}

3、只拦截目标类的目标方法(其他普通方法不拦截)的逻辑在哪实现,如何调用到Interceptor的interceptor方法?
上面Plugin类中的invoke方法执行时,会对方法进行判断是否需要增强,然后执行interceptor.intercept

总结

mybatis 的插件原理其实还是比较简单的,对四大组件指定的方法实现方法增强,多个拦截器就会创建多个代理对象,一层层包装,形成拦截器链,逐个执行。
另:
四大组件一般就Executor(最常见) 和StatementHandler用的多一些,mybatis-plus中
MybatisPlusInterceptor对Executor的query,update,StatementHandler的prepare 方法都进行了拦截,在其中设置了内部拦截器(InnerInterceptor),在不同执行时机都预留了接口,实现拦截器时直接都是实现内部拦截器,这样就不需要创建多个代理对象,性能应该会好一点。