引用自官網(wǎng):XML 映射配置文件
MyBatis 允許你在已映射語句執(zhí)行過程中的某一點進行攔截調(diào)用揭蜒。默認情況下厉碟,MyBatis 允許使用插件來攔截的方法調(diào)用包括:
- Executor (update, query, flushStatements, commit, rollback, getTransaction, close, isClosed)
- ParameterHandler (getParameterObject, setParameters)
- ResultSetHandler (handleResultSets, handleOutputParameters)
- StatementHandler (prepare, parameterize, batch, update, query)
這些類中方法的細節(jié)可以通過查看每個方法的簽名來發(fā)現(xiàn)毙石,或者直接查看 MyBatis 的發(fā)行包中的源代碼纺座。 假設(shè)你想做的不僅僅是監(jiān)控方法的調(diào)用碌秸,那么你應(yīng)該很好的了解正在重寫的方法的行為撤卢。 因為如果在試圖修改或重寫已有方法的行為的時候素征,你很可能在破壞 MyBatis 的核心模塊集嵌。 這些都是更低層的類和方法,所以使用插件的時候要特別當心御毅。
通過 MyBatis 提供的強大機制根欧,使用插件是非常簡單的,只需實現(xiàn) Interceptor 接口端蛆,并指定了想要攔截的方法簽名即可凤粗。
1. 使用插件與不使用插件(Mybatis自身日志記錄的SQL)打印SQL的區(qū)別
以下就是使用Mybatis自身日志記錄SQL所存在的問題
- SQL中參數(shù)都被占位符"?"替換,無法知道真正執(zhí)行的SQL語句中的參數(shù)是什么
- 無法記錄SQL執(zhí)行時間今豆,有SQL執(zhí)行時間就可以精準定位到執(zhí)行時間比較慢的SQL
2. 實現(xiàn)SQL攔截器
攔截SQL只需要實現(xiàn)
org.apache.ibatis.plugin.Interceptor
接口即可:
package com.batatas.framework.mybatis.plugin;
import java.lang.reflect.Field;
import java.sql.Statement;
import java.util.Collection;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import org.apache.ibatis.executor.statement.StatementHandler;
import org.apache.ibatis.mapping.BoundSql;
import org.apache.ibatis.mapping.ParameterMapping;
import org.apache.ibatis.plugin.Interceptor;
import org.apache.ibatis.plugin.Intercepts;
import org.apache.ibatis.plugin.Invocation;
import org.apache.ibatis.plugin.Plugin;
import org.apache.ibatis.plugin.Signature;
import org.apache.ibatis.session.ResultHandler;
import org.apache.ibatis.session.defaults.DefaultSqlSession.StrictMap;
@Intercepts(value = {
@Signature(args = { Statement.class, ResultHandler.class }, method = "query", type = StatementHandler.class),
@Signature(args = { Statement.class }, method = "update", type = StatementHandler.class),
@Signature(args = { Statement.class }, method = "batch", type = StatementHandler.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語句嫌拣,去除換行符柔袁,替換參數(shù)
sql = formatSql(sql, parameterObject, parameterMappingList);
System.out.println("SQL:[" + sql + "]執(zhí)行耗時[" + sqlCost + "ms]");
}
}
@Override
public Object plugin(Object target) {
return Plugin.wrap(target, this);
}
@Override
public void setProperties(Properties properties) {
}
private String formatSql(String sql, Object parameterObject, List<ParameterMapping> parameterMappingList) {
// 輸入判斷是否為空
if (sql == "" || sql.length() == 0) {
return "";
}
// 美化sql
sql = beautifySql(sql);
// 不傳參數(shù)的場景,直接把Sql美化一下返回出去
if (parameterObject == null || parameterMappingList == null || parameterMappingList.size() == 0) {
return sql;
}
// 定義一個沒有替換過占位符的sql亭罪,用于出異常時返回
String sqlWithoutReplacePlaceholder = sql;
try {
if (parameterMappingList != null) {
Class<?> parameterObjectClass = parameterObject.getClass();
// 如果參數(shù)是StrictMap且Value類型為Collection瘦馍,獲取key="list"的屬性,這里主要是為了處理<foreach>循環(huán)時傳入List這種參數(shù)的占位符替換
// 例如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)) {
// 如果參數(shù)是Map則直接強轉(zhuǎn)应役,通過map.get(key)方法獲取真正的屬性值
// 這里主要是為了處理<insert>情组、<delete>、<update>箩祥、<select>時傳入parameterType為map的場景
Map<?, ?> paramMap = (Map<?, ?>) parameterObject;
sql = handleMapParameter(sql, paramMap, parameterMappingList);
} else {
// 通用場景院崇,比如傳的是一個自定義的對象或者八種基本數(shù)據(jù)類型之一或者String
sql = handleCommonParameter(sql, parameterMappingList, parameterObjectClass, parameterObject);
}
}
} catch (Exception e) {
// 占位符替換過程中出現(xiàn)異常,則返回沒有替換過占位符但是格式美化過的sql袍祖,這樣至少保證sql語句比BoundSql中的sql更好看
return sqlWithoutReplacePlaceholder;
}
return sql;
}
/**
* 處理通用場景
*
* @throws SecurityException
* @throws NoSuchFieldException
* @throws IllegalAccessException
* @throws IllegalArgumentException
*/
private String handleCommonParameter(String sql, List<ParameterMapping> parameterMappingList,
Class<?> parameterObjectClass, Object parameterObject) throws Exception {
for (ParameterMapping parameterMapping : parameterMappingList) {
String propertyValue = null;
// 基本數(shù)據(jù)類型或者基本數(shù)據(jù)類型的包裝類底瓣,直接toString即可獲取其真正的參數(shù)值,其余直接取paramterMapping中的property屬性即可
if (isPrimitiveOrPrimitiveWrapper(parameterObjectClass)) {
propertyValue = parameterObject.toString();
} else {
String propertyName = parameterMapping.getProperty();
Field field = parameterObjectClass.getDeclaredField(propertyName);
// 要獲取Field中的屬性值蕉陋,這里必須將私有屬性的accessible設(shè)置為true
field.setAccessible(true);
propertyValue = String.valueOf(field.get(parameterObject));
if (parameterMapping.getJavaType().isAssignableFrom(String.class)) {
propertyValue = "\"" + propertyValue + "\"";
}
}
sql = sql.replaceFirst("\\?", propertyValue);
}
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;
}
/**
* @Description: 處理List場景
* @param sql
* @param collection
*/
private String handleListParameter(String sql, Collection<?> col) {
if (col != null && col.size() != 0) {
for (Object obj : col) {
String value = null;
Class<?> objClass = obj.getClass();
// 只處理基本數(shù)據(jù)類型捐凭、基本數(shù)據(jù)類型的包裝類、String這三種
// 如果是復(fù)合類型也是可以的凳鬓,不過復(fù)雜點且這種場景較少茁肠,寫代碼的時候要判斷一下要拿到的是復(fù)合類型中的哪個屬性
if (isPrimitiveOrPrimitiveWrapper(objClass)) {
value = obj.toString();
} else if (objClass.isAssignableFrom(String.class)) {
value = "\"" + obj.toString() + "\"";
}
sql = sql.replaceFirst("\\?", value);
}
}
return sql;
}
private String beautifySql(String sql) {
// sql = sql.replace("\n", "").replace("\t", "").replace(" ", "
// ").replace("( ", "(").replace(" )", ")").replace(" ,", ",");
sql = sql.replaceAll("[\\s\n ]+", " ");
return sql;
}
/**
* 是否基本數(shù)據(jù)類型或者基本數(shù)據(jù)類型的包裝類
*/
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的內(nèi)部類StrictMap
*/
private boolean isStrictMap(Class<?> parameterObjectClass) {
return parameterObjectClass.isAssignableFrom(StrictMap.class);
}
/**
* 是否List的實現(xiàn)類
*/
private boolean isList(Class<?> clazz) {
Class<?>[] interfaceClasses = clazz.getInterfaces();
for (Class<?> interfaceClass : interfaceClasses) {
if (interfaceClass.isAssignableFrom(List.class)) {
return true;
}
}
return false;
}
/**
* 是否Map的實現(xiàn)類
*/
private boolean isMap(Class<?> parameterObjectClass) {
Class<?>[] interfaceClasses = parameterObjectClass.getInterfaces();
for (Class<?> interfaceClass : interfaceClasses) {
if (interfaceClass.isAssignableFrom(Map.class)) {
return true;
}
}
return false;
}
}
3. 配置插件
配置插件,只需要在mybatis-config.xml文件中添加插件即可:
<!-- mybatis-config.xml -->
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE configuration
PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
<plugins>
<plugin interceptor="com.batatas.framework.mybatis.plugin.SqlCostInterceptor">
<!-- <property name="prop1" value="prop1"/>
<property name="prop2" value="prop2"/> -->
</plugin>
</plugins>
</configuration>