【MyBatis源码分析】plugins解析属性配置元素详述

Plugins解析、

plugin有何作用? 需要配置吗?

  plugins 是一个可选配置。mybatis中的plugin其实就是个interceptor, 它可以拦截Executor 、ParameterHandler 、ResultSetHandler 、StatementHandler 的部分方法,处理我们自己的逻辑。Executor就是真正执行sql语句的东西, ParameterHandler 是处理我们传入参数的,还记得前面讲TypeHandler的时候提到过,mybatis默认帮我们实现了不少的typeHandler, 当我们不显示配置typeHandler的时候,mybatis会根据参数类型自动选择合适的typeHandler执行,其实就是ParameterHandler 在选择。ResultSetHandler 就是处理返回结果的。由于MyBatis的插件已经深入到了MyBatis底层代码,因此要更好地使用插件,必须对插件实现原理及MyBatis底层代码有所熟悉才行,本文分析一下MyBatis的插件实现原理。

   怎么自定义plugin ? 怎么配置?

   要自定义一个plugin, 需要去实现Interceptor接口。定义好之后,配置如下:

<plugins>
		<plugin interceptor="Interceptor.SqlCostInterceptor"/>
     </plugins>
接下来我们看实现一个SqlCostInterceptor类;

package Interceptor;

import java.util.Collection;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.lang.reflect.Field;
import java.sql.Statement;

import org.apache.ibatis.executor.statement.StatementHandler;
import org.apache.ibatis.mapping.BoundSql;
import org.apache.ibatis.mapping.ParameterMapping;
import org.apache.ibatis.plugin.Signature;
import org.apache.ibatis.session.ResultHandler;
import org.apache.ibatis.session.defaults.DefaultSqlSession.StrictMap;
import org.apache.ibatis.plugin.Interceptor;
import org.apache.ibatis.plugin.Intercepts;
import org.apache.ibatis.plugin.Invocation;
import org.apache.ibatis.plugin.Plugin;

/**
  * Sql执行时间记录拦截器 
  */
@Intercepts(
			{@Signature(type = StatementHandler.class, method = "query", args = {Statement.class, ResultHandler.class}),
			@Signature(type = StatementHandler.class, method = "update", args = {Statement.class}),
			@Signature(type = StatementHandler.class, method = "batch", args = { Statement.class })})
public class SqlCostInterceptor implements Interceptor {
	
	@Override
	public Object intercept(Invocation invocation) throws Throwable {
	     Object target = invocation.getTarget();
	     
	     long startTime = System.currentTimeMillis();
	     StatementHandler statementHandler = (StatementHandler)target;
	     try {
	         return invocation.proceed();
	     } finally {
	         long endTime = System.currentTimeMillis();
	         long sqlCost = endTime - startTime;
	         
	         BoundSql boundSql = statementHandler.getBoundSql();
	         String sql = boundSql.getSql();
	         Object parameterObject = boundSql.getParameterObject();
	         List<ParameterMapping> parameterMappingList = boundSql.getParameterMappings();
	         // 格式化Sql语句,去除换行符,替换参数
	         sql = formatSql(sql, parameterObject, parameterMappingList);
	         System.out.println("SQL:[" + sql + "]执行耗时[" + sqlCost + "ms]");
	     }
	}
	
	@Override
	public Object plugin(Object target) {
	     return Plugin.wrap(target, this);
	}
	
	@Override
	public void setProperties(Properties properties) {
	       
	}
	
	
	@SuppressWarnings("unchecked")
	private String formatSql(String sql, Object parameterObject, List<ParameterMapping> parameterMappingList) {
	 // 输入sql字符串空判断
	   if (sql == null || sql.length() == 0) {
	          return "";
	     }
	  
	    // 美化sql
	    sql = beautifySql(sql);
	  
	   // 不传参数的场景,直接把Sql美化一下返回出去
	   if (parameterObject == null || parameterMappingList == null || parameterMappingList.size() == 0) {
	      return sql;
	     }
	     
	      // 定义一个没有替换过占位符的sql,用于出异常时返回
	     String sqlWithoutReplacePlaceholder = sql;
	   
	     try {
	      if (parameterMappingList != null) {
	              Class<?> parameterObjectClass = parameterObject.getClass();
	
	           // 如果参数是StrictMap且Value类型为Collection,获取key="list"的属性,这里主要是为了处理<foreach>循环时传入List这种参数的占位符替换
	   // 例如select * from xxx where id in <foreach collection="list">...</foreach>
	if (isStrictMap(parameterObjectClass)) {
	  StrictMap<Collection<?>> strictMap = (StrictMap<Collection<?>>)parameterObject;
	     
	   if (isList(strictMap.get("list").getClass())) {
	sql = handleListParameter(sql, strictMap.get("list"));
	 }
	} else if (isMap(parameterObjectClass)) {
	   // 如果参数是Map则直接强转,通过map.get(key)方法获取真正的属性值
	  // 这里主要是为了处理<insert>、<delete>、<update>、<select>时传入parameterType为map的场景
	      Map<?, ?> paramMap = (Map<?, ?>) parameterObject;
	   sql = handleMapParameter(sql, paramMap, parameterMappingList);
	} else {
	    // 通用场景,比如传的是一个自定义的对象或者八种基本数据类型之一或者String
	             sql = handleCommonParameter(sql, parameterMappingList, parameterObjectClass, parameterObject);
	           }
	       }
	 } catch (Exception e) {
	       // 占位符替换过程中出现异常,则返回没有替换过占位符但是格式美化过的sql,这样至少保证sql语句比BoundSql中的sql更好看
	            return sqlWithoutReplacePlaceholder;
	       }
	         return sql;
	}
	
	/**
	* 美化Sql
	*/
	private String beautifySql(String sql) {
	   // sql = sql.replace("\n", "").replace("\t", "").replace("  ", " ").replace("( ", "(").replace(" )", ")").replace(" ,", ",");
	   sql = sql.replaceAll("[\\s\n ]+"," ");
	   return sql;
	}
	
	
	/**
	* 处理参数为List的场景
	*/
	private String handleListParameter(String sql, Collection<?> col) {
	     if (col != null && col.size() != 0) {
	      for (Object obj : col) {
	        String value = null;
	         Class<?> objClass = obj.getClass();
	         
	            // 只处理基本数据类型、基本数据类型的包装类、String这三种
	 // 如果是复合类型也是可以的,不过复杂点且这种场景较少,写代码的时候要判断一下要拿到的是复合类型中的哪个属性
	 if (isPrimitiveOrPrimitiveWrapper(objClass)) {
	    value = obj.toString();
	    } else if (objClass.isAssignableFrom(String.class)) {
	             value = "\"" + obj.toString() + "\""; 
	   }
	         
	   sql = sql.replaceFirst("\\?", value);
	       }
	    }
	         
	  return sql;
	}
	
	
	 /**
	* 处理参数为Map的场景
	*/
	private String handleMapParameter(String sql, Map<?, ?> paramMap, List<ParameterMapping> parameterMappingList) {
	 for (ParameterMapping parameterMapping : parameterMappingList) {
	   Object propertyName = parameterMapping.getProperty();
	           Object propertyValue = paramMap.get(propertyName);
	            if (propertyValue != null) {
	                if (propertyValue.getClass().isAssignableFrom(String.class)) {
	                 propertyValue = "\"" + propertyValue + "\"";
	                }
	 
	                 sql = sql.replaceFirst("\\?", propertyValue.toString());
	         }
	     }
	     
	     return sql;
	 }
	
	
	/**
	      * 处理通用的场景
	      */
	 private String handleCommonParameter(String sql, List<ParameterMapping> parameterMappingList, Class<?> parameterObjectClass, 
	         Object parameterObject) throws Exception {
	     for (ParameterMapping parameterMapping : parameterMappingList) {
	         String propertyValue = null;
	         // 基本数据类型或者基本数据类型的包装类,直接toString即可获取其真正的参数值,其余直接取paramterMapping中的property属性即可
	 if (isPrimitiveOrPrimitiveWrapper(parameterObjectClass)) {
	  propertyValue = parameterObject.toString();
	} else {
	     String propertyName = parameterMapping.getProperty();
	     
	    Field field = parameterObjectClass.getDeclaredField(propertyName);
	    // 要获取Field中的属性值,这里必须将私有属性的accessible设置为true
	    field.setAccessible(true);
	    propertyValue = String.valueOf(field.get(parameterObject));
	  if (parameterMapping.getJavaType().isAssignableFrom(String.class)) {
	         propertyValue = "\"" + propertyValue + "\"";
	   }
	 }
	
	sql = sql.replaceFirst("\\?", propertyValue);
	        }
	        
	       return sql;
	    }
	
	 /**
	* 是否基本数据类型或者基本数据类型的包装类
	 */
	   private boolean isPrimitiveOrPrimitiveWrapper(Class<?> parameterObjectClass) {
	         return parameterObjectClass.isPrimitive() || 
	                 (parameterObjectClass.isAssignableFrom(Byte.class) || parameterObjectClass.isAssignableFrom(Short.class) ||
	                         parameterObjectClass.isAssignableFrom(Integer.class) || parameterObjectClass.isAssignableFrom(Long.class) ||
	                        parameterObjectClass.isAssignableFrom(Double.class) || parameterObjectClass.isAssignableFrom(Float.class) ||
	                     parameterObjectClass.isAssignableFrom(Character.class) || parameterObjectClass.isAssignableFrom(Boolean.class));
	   }
	 
	   /**
	* 是否DefaultSqlSession的内部类StrictMap
	 */
	   private boolean isStrictMap(Class<?> parameterObjectClass) {
	    return parameterObjectClass.isAssignableFrom(StrictMap.class);
	    }
	   
	     /**
	* 是否List的实现类
	 */
	    private boolean isList(Class<?> clazz) {
	        Class<?>[] interfaceClasses = clazz.getInterfaces();
	        for (Class<?> interfaceClass : interfaceClasses) {
	           if (interfaceClass.isAssignableFrom(List.class)) {
	               return true;
	         }
	       }
	        
	       return false;
	   }
	   
	/**
	     * 是否Map的实现类
	 */
	     private boolean isMap(Class<?> parameterObjectClass) {
	     Class<?>[] interfaceClasses = parameterObjectClass.getInterfaces();
	       for (Class<?> interfaceClass : interfaceClasses) {
	            if (interfaceClass.isAssignableFrom(Map.class)) {
	                 return true;
	           }
	 }
	         
	return false;
	}
}

接着看一下pluginElement(root.evalNode("plugins"));方法,这句读取的是<configuration>下的<plugins>节点,代码实现为:

private void pluginElement(XNode parent) throws Exception {
    if (parent != null) {
      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);
      }
    }
  }

这里拿<plugin>标签中的interceptor属性,这是自定义的拦截器的全路径,第6行的代码通过反射生成拦截器实例。

再拿<plugin>标签下的所有<property>标签,解析name和value属性成为一个Properties,将Properties设置到拦截器中。

最后,通过第8行的代码将拦截器设置到Configuration中,源码实现为:

public void addInterceptor(Interceptor interceptor) {
    interceptorChain.addInterceptor(interceptor);
  }
InterceptorChain是一个拦截器链,存储了所有定义的拦截器以及相关的几个操作的方法:
public class InterceptorChain {

  private final List<Interceptor> interceptors = new ArrayList<Interceptor>();

  public Object pluginAll(Object target) {
    for (Interceptor interceptor : interceptors) {
      target = interceptor.plugin(target);
    }
    return target;
  }

  public void addInterceptor(Interceptor interceptor) {
    interceptors.add(interceptor);
  }
  
  public List<Interceptor> getInterceptors() {
    return Collections.unmodifiableList(interceptors);
  }

}
分别有添加拦截器、为目标对象添加所有拦截器、获取当前所有拦截器三个方法。

MyBatis插件原理----pluginAll方法添加插件

上面我们在InterceptorChain中看到了一个pluginAll方法,pluginAll方法为目标对象生成代理,之后目标对象调用方法的时候走的不是原方法而是代理方法,这个在后面会说明。

MyBatis官网文档有说明,在以下四个代码执行点上允许使用插件:


为之生成插件的时机(换句话说就是pluginAll方法调用的时机)是Executor、ParameterHandler、ResultSetHandler、StatementHandler四个接口实现类生成的时候,每个接口实现类在MyBatis中生成的时机是不一样的,这个就不看它们是在什么时候生成的了,每个开发工具我相信都有快捷键可以看到pluginAll方法调用的地方,我使用的Eclipse就是Ctrl+Alt+H。

再看pluginAll方法:

public Object pluginAll(Object target) {
    for (Interceptor interceptor : interceptors) {
      target = interceptor.plugin(target);
    }
    return target;
  }
这里值得注意的是:

  1. 形参Object target,这个是Executor、ParameterHandler、ResultSetHandler、StatementHandler接口的实现类,换句话说,plugin方法是要为Executor、ParameterHandler、ResultSetHandler、StatementHandler的实现类生成代理,从而在调用这几个类的方法的时候,其实调用的是InvocationHandler的invoke方法
  2. 这里的target是通过for循环不断赋值的,也就是说如果有多个拦截器,那么如果我用P表示代理,生成第一次代理为P(target),生成第二次代理为P(P(target)),生成第三次代理为P(P(P(target))),不断嵌套下去,这就得到一个重要的结论:<plugins>...</plugins>中后定义的<plugin>实际其拦截器方法先被执行,因为根据这段代码来看,后定义的<plugin>代理实际后生成,包装了先生成的代理,自然其代理方法也先执行(先进后出<栈>)
plugin方法中调用MyBatis提供的现成的生成代理的方法Plugin.wrap(Object target, Interceptor interceptor),接着我们看下wrap方法的源码实现。

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;
  }
首先看一下第2行的代码,获取Interceptor上定义的所有方法签名:
private static Map<Class<?>, Set<Method>> getSignatureMap(Interceptor interceptor) {
    Intercepts interceptsAnnotation = interceptor.getClass().getAnnotation(Intercepts.class);
    // issue #251
    if (interceptsAnnotation == null) {
      throw new PluginException("No @Intercepts annotation was found in interceptor " + interceptor.getClass().getName());      
    }
    Signature[] sigs = interceptsAnnotation.value();
    Map<Class<?>, Set<Method>> signatureMap = new HashMap<Class<?>, Set<Method>>();
    for (Signature sig : sigs) {
      Set<Method> methods = signatureMap.get(sig.type());
      if (methods == null) {
        methods = new HashSet<Method>();
        signatureMap.put(sig.type(), methods);
      }
      try {
        Method method = sig.type().getMethod(sig.method(), sig.args());
        methods.add(method);
      } catch (NoSuchMethodException e) {
        throw new PluginException("Could not find method on " + sig.type() + " named " + sig.method() + ". Cause: " + e, e);
      }
    }
    return signatureMap;
  }

看到先拿@Intercepts注解,如果没有定义@Intercepts注解,抛出异常,这意味着使用MyBatis的插件,必须使用注解方式

接着拿到@Intercepts注解下的所有@Signature注解,获取其type属性(表示具体某个接口),再根据method与args两个属性去type下找方法签名一致的方法Method(如果没有方法签名一致的就抛出异常,此签名的方法在该接口下找不到),能找到的话key=type,value=Set<Method>,添加到signatureMap中,构建出一个方法签名映射。举个例子来说,就是我定义的@Intercepts注解,Executor下我要拦截的所有Method、StatementHandler下我要拦截的所有Method。

回过头继续看wrap方法,在拿到方法签名映射后,调用getAllInterfaces方法,传入的是Target的Class对象以及之前获取到的方法签名映射:

private static Class<?>[] getAllInterfaces(Class<?> type, Map<Class<?>, Set<Method>> signatureMap) {
    Set<Class<?>> interfaces = new HashSet<Class<?>>();
    while (type != null) {
      for (Class<?> c : type.getInterfaces()) {
        if (signatureMap.containsKey(c)) {
          interfaces.add(c);
        }
      }
      type = type.getSuperclass();
    }
    return interfaces.toArray(new Class<?>[interfaces.size()]);
  }
这里获取Target的所有接口,如果方法签名映射中有这个接口,那么添加到interfaces中,这是一个Set,最终将Set转换为数组返回。
wrap方法的最后一步:
if (interfaces.length > 0) {
      return Proxy.newProxyInstance(
          type.getClassLoader(),
          interfaces,
          new Plugin(target, interceptor, signatureMap));
    }
    return target;

果当前传入的Target的接口中有@Intercepts注解中定义的接口,那么为之生成代理,否则原Target返回。

这段理论可能大家会看得有点云里雾里,我这里举个例子:

就以SqlCostPlugin为例,我的@Intercepts定义的是:
@Intercepts({@Signature(type = StatementHandler.class, method = "query", args = {Statement.class, ResultHandler.class}), @Signature(type = StatementHandler.class, 
method = "update", args = {Statement.class})})

此时,生成的方法签名映射signatureMap应当是(我这里把Map给toString()了):
{interface org.apache.ibatis.executor.statement.StatementHandler=[public abstract int org.apache.ibatis.executor.statement.StatementHandler.update(java.sql.
Statement) throws java.sql.SQLException, public abstract java.util.List org.apache.ibatis.executor.statement.StatementHandler.query(java.sql.Statement,org.apache.
ibatis.session.ResultHandler) throws java.sql.SQLException]}
一个Class对应一个Set,Class为StatementHandler.class,Set为StataementHandler中的两个方法

如果我new的是StatementHandler接口的实现类,那么可以为之生成代理,因为signatureMap中的key有StatementHandler这个接口

如果我new的是Executor接口的实现类,那么直接会把Executor接口的实现类原样返回,因为signatureMap中的key并没有Executor这个接口
相信这么解释大家应该会明白一点。注意这里生不生成代理,只和接口在不在@Intercepts中定义过有关,和方法签名无关,具体某个方法走拦截器,在invoke方法中,马上来看一下。

MyBatis插件原理----Plugin的invoke方法

首先看一下Plugin方法的方法定义:

public class Plugin implements InvocationHandler {

  private Object target;
  private Interceptor interceptor;
  private Map<Class<?>, Set<Method>> signatureMap;

  private Plugin(Object target, Interceptor interceptor, Map<Class<?>, Set<Method>> signatureMap) {
    this.target = target;
    this.interceptor = interceptor;
    this.signatureMap = signatureMap;
  }
  ...
}
看到Plugin是InvocationHandler接口的实现类,换句话说,为目标接口生成代理之后,最终执行的都是Plugin的invoke方法,看一下invoke方法的实现:
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);
    }
}

在这里,将method对应的Class拿出来,获取该Class中有哪些方法签名,换句话说就是Executor、ParameterHandler、ResultSetHandler、StatementHandler,在@Intercepts注解中定义了要拦截哪些方法签名。

如果当前调用的方法的方法签名在方法签名集合中,即满足第4行的判断,那么调用拦截器的intercept方法,否则方法原样调用,不会执行拦截器。










  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值