使用Mybatis外掛列印SQL詳細內容及執行時間
引用自官網:XML 對映配置檔案
MyBatis 允許你在已對映語句執行過程中的某一點進行攔截呼叫。預設情況下,MyBatis 允許使用外掛來攔截的方法呼叫包括:
- Executor (update, query, flushStatements, commit, rollback, getTransaction, close, isClosed)
- ParameterHandler (getParameterObject, setParameters)
- ResultSetHandler (handleResultSets, handleOutputParameters)
- StatementHandler (prepare, parameterize, batch, update, query)
這些類中方法的細節可以通過檢視每個方法的簽名來發現,或者直接檢視 MyBatis 的發行包中的原始碼。 假設你想做的不僅僅是監控方法的呼叫,那麼你應該很好的瞭解正在重寫的方法的行為。 因為如果在試圖修改或重寫已有方法的行為的時候,你很可能在破壞 MyBatis 的核心模組。 這些都是更低層的類和方法,所以使用外掛的時候要特別當心。
通過 MyBatis 提供的強大機制,使用外掛是非常簡單的,只需實現 Interceptor 介面,並指定了想要攔截的方法簽名即可。
1. 使用外掛與不使用外掛(Mybatis自身日誌記錄的SQL)列印SQL的區別
以下就是使用Mybatis自身日誌記錄SQL所存在的問題
- SQL中引數都被佔位符"?"替換,無法知道真正執行的SQL語句中的引數是什麼
- 無法記錄SQL執行時間,有SQL執行時間就可以精準定位到執行時間比較慢的SQL
2. 實現SQL攔截器
攔截SQL只需要實現
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語句,去除換行符,替換引數
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) {
}
private String formatSql(String sql, Object parameterObject, List<ParameterMapping> parameterMappingList) {
// 輸入判斷是否為空
if (sql == "" || 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;
}
/**
* 處理通用場景
*
* @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;
// 基本資料型別或者基本資料型別的包裝類,直接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;
}
/**
* 處理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();
// 只處理基本資料型別、基本資料型別的包裝類、String這三種
// 如果是複合型別也是可以的,不過複雜點且這種場景較少,寫程式碼的時候要判斷一下要拿到的是複合型別中的哪個屬性
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;
}
/**
* 是否基本資料型別或者基本資料型別的包裝類
*/
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;
}
}
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>
4. 測試
相關文章
- MyBatis列印SQL執行時間MyBatisSQL
- MyBatis 核心配置檔案詳細內容詳解MyBatis
- laravel列印輸出完整sql,執行時間和explain分析LaravelSQLAI
- 記錄一次mybatis log-plugin外掛不列印sqlMyBatisPluginSQL
- 七,MyBatis-Plus 擴充套件功能:樂觀鎖,程式碼生成器,執行SQL分析列印(實操詳細使用)MyBatis套件SQL
- MyBatis SQL執行MyBatisSQL
- oracle查詢sql執行耗時、執行時間、sql_idOracleSQL
- onethink 如何使用外掛 詳細的教程
- MyBatis(八):MyBatis外掛機制詳解MyBatis
- laravel Modules外掛內定時任務執行,自定義命令註冊,外掛內資源釋出Laravel
- MyBatis逆向工 maven外掛 generator 的配置及使用MyBatisMaven
- MyBatis 關於查詢語句上配置的詳細內容MyBatis
- maven外掛執行過程中自動執行sql檔案MavenSQL
- mybatis執行sql指令碼MyBatisSQL指令碼
- Mybatis 列印完整的SQLMyBatisSQL
- 疑問:mybatis如何自定義SQL執行時長MyBatisSQL
- Yii筆記:列印sql、Form表單、時間外掛、Mysql的 FIND_IN_SET函式使用、是否是post/ajax請求筆記ORMMySql函式
- Yii1列印當前請求所有執行的SQL及耗時SQL
- 檢視谷歌seo內容外掛谷歌
- MyBatis外掛MyBatis
- 深入淺出MyBatis:MyBatis外掛及開發過程MyBatis
- 使用sql monitor獲取更加詳細的執行計劃 - dbms_sqltune.report_sql_monitorSQL
- [外掛擴充套件]跪求時間軸外掛套件
- sublime安裝外掛詳細教程
- goby 外掛推薦 及 內網下載使用Go內網
- Maven外掛執行方式Maven
- MyBatis Generator配置及執行MyBatis
- 列印外掛工作原理
- mybatis原始碼學習:外掛定義+執行流程責任鏈MyBatis原始碼
- PHP執行耗時指令碼實時輸出內容PHP指令碼
- @軟考生,准考證列印時間已出,請及時列印
- Mybatis 動態執行SQL語句MyBatisSQL
- 前端meta標籤內容定義及使用說明,meta詳細說明,meta標籤使用前端
- 獲取oracle sql語句詳細些執行計劃OracleSQL
- mybatis generator外掛系列--分頁外掛MyBatis
- Eclipse安裝lombok外掛及外掛使用案例EclipseLombok
- mybatis plus 啟用 mybatis外掛MyBatis
- Linux /etc/shadow 超詳細內容解析Linux
- 自研WPF外掛系統(沙箱執行及熱插拔)