打印SQL的执行时间,我们可以实现mybatis官方我们提供的接口org.apache.ibatis.plugin.Interceptor,我们可以拦截的类有多个Executor,StatementHandler,ParameterHandler等,第一次写拦截的时候选择了Executor,但是我发现有些SQL拦截了之后是找不到具体的参数并填充到SQL中(原因是在我们拦截了之后框架又做了一次拦截,导致我们拦截的不是最终的SQL),所以我最后查阅了mybatis源码,捋清楚了SQL的执行流程,我选择了拦截StatementHandler,原因就是StatementHandler是在SQL执行的流程的最后面,这样,不管前面是否有人拦截了SQL,我们都可以正常的获取到SQL并且获取到对应的参数,填充到SQL中。

下面附上mybatis的执行流程图:

具体的代码如下:

SqlCostPlugins.java



import org.apache.ibatis.executor.parameter.ParameterHandler;
import org.apache.ibatis.executor.statement.StatementHandler;
import org.apache.ibatis.mapping.BoundSql;
import org.apache.ibatis.mapping.MappedStatement;
import org.apache.ibatis.mapping.ParameterMapping;
import org.apache.ibatis.mapping.ParameterMode;
import org.apache.ibatis.plugin.*;
import org.apache.ibatis.reflection.MetaObject;
import org.apache.ibatis.session.Configuration;
import org.apache.ibatis.session.ResultHandler;
import org.apache.ibatis.type.TypeHandlerRegistry;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.lang.reflect.Field;
import java.sql.Statement;
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.time.ZoneId;
import java.time.format.DateTimeFormatter;
import java.util.Date;
import java.util.List;
import java.util.Objects;
import java.util.Properties;

/**
* <p>
* 自定义SQL插件,功能如下
* 1:打印SQL执行时间
* 2:打印SQL,参数自动设置到SQL中
* 3:区别慢SQL,SQL执行时间大于5秒的SQL为红色字体,否则为黄色字体,(执行时间可以自定义)
* </p>
*
* @author liekkas 2020/12/08 10:42
*/
@Intercepts({@Signature(type = StatementHandler.class, method = "update", args = {Statement.class,}),
@Signature(type = StatementHandler.class, method = "query", args = {Statement.class, ResultHandler.class})})
public final class SqlCostPlugins implements Interceptor {

private static final DateTimeFormatter DATE_TIME_FORMATTER = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");
private static final DateTimeFormatter DATE_FORMATTER = DateTimeFormatter.ofPattern("yyyy-MM-dd");

private static final Logger LOGGER = LoggerFactory.getLogger(SqlCostPlugins.class);

@Override
public Object intercept(Invocation invocation) throws Throwable {
long startTime = System.currentTimeMillis();
try {
return invocation.proceed();
} finally {
long endTime = System.currentTimeMillis();
long elapsedTime = endTime - startTime;
String sql = formatSql(invocation);
printColorString(String.format("cost %s ms,执行SQL:\n %s ", elapsedTime, sql), elapsedTime);
}
}

@Override
public Object plugin(Object target) {
return Plugin.wrap(target, this);
}

@Override
public void setProperties(Properties properties) {

}

/**
* 格式化SQL及其参数
*
* @param invocation invocation
* @return java.lang.String
* @author liekkas 2020/12/08 10:43
*/
private String formatSql(Invocation invocation) throws NoSuchFieldException, IllegalAccessException {
//获取StatementHandler
StatementHandler statementHandler = (StatementHandler) invocation.getTarget();
//获取ParameterHandler
ParameterHandler parameterHandler = statementHandler.getParameterHandler();
//获取boundSql
BoundSql boundSql = statementHandler.getBoundSql();

Class<? extends ParameterHandler> parameterHandlerClass = parameterHandler.getClass();
Field mappedStatementField = parameterHandlerClass.getDeclaredField("mappedStatement");
mappedStatementField.setAccessible(true);
MappedStatement mappedStatement = (MappedStatement) mappedStatementField.get(parameterHandler);

String sql = boundSql.getSql();

// 输入sql字符串空判断
if (Objects.isNull(sql)) {
return "";
}

// 美化sql
sql = beautifySql(sql).toLowerCase();

// 不传参数的场景,直接把Sql美化一下返回出去
Object parameterObject = parameterHandler.getParameterObject();
List<ParameterMapping> parameterMappingList = boundSql.getParameterMappings();
if (Objects.isNull(parameterObject) || parameterMappingList.isEmpty()) {
return sql;
}

// 定义一个没有替换过占位符的sql,用于出异常时返回
String sqlWithoutReplacePlaceholder = sql;

try {
sql = handleCommonParameter(boundSql, mappedStatement);
} catch (Exception e) {
System.err.println(e.getMessage());
// 占位符替换过程中出现异常,则返回没有替换过占位符但是格式美化过的sql
return sqlWithoutReplacePlaceholder;
}

return sql;
}

/**
* 美化SQL
*
* @param sql sql
* @return java.lang.String
* @author liekkas 2020/12/08 10:45
*/
private String beautifySql(String sql) {
sql = sql.replaceAll("[\\s\n ]+", " ");
return sql;
}

/**
* 替换SQL中的?,设置sql参数
*
* @param boundSql boundSql
* @param mappedStatement mappedStatement
* @return java.lang.String
* @author liekkas 2020/12/08 10:46
*/
private String handleCommonParameter(BoundSql boundSql, MappedStatement mappedStatement) {

String sql = boundSql.getSql();
Object parameterObject = boundSql.getParameterObject();
List<ParameterMapping> parameterMappings = boundSql.getParameterMappings();
Configuration configuration = mappedStatement.getConfiguration();
TypeHandlerRegistry typeHandlerRegistry = configuration.getTypeHandlerRegistry();

for (ParameterMapping parameterMapping : parameterMappings) {
if (parameterMapping.getMode() != ParameterMode.OUT) {
Object value;
String propertyName = parameterMapping.getProperty();
if (boundSql.hasAdditionalParameter(propertyName)) {
value = boundSql.getAdditionalParameter(propertyName);
} else if (parameterObject == null) {
value = null;
} else if (typeHandlerRegistry.hasTypeHandler(parameterObject.getClass())) {
value = parameterObject;
} else {
MetaObject metaObject = configuration.newMetaObject(parameterObject);
value = metaObject.getValue(propertyName);
}
sql = replacePlaceholder(sql, value);
}
}
return sql;
}

/**
* 根据不同的propertyValue类型,匹配SQL?的类型并替换值
*
* @param sql sql
* @param propertyValue propertyValue
* @return java.lang.String
* @author liekkas 2020/12/08 10:48
*/
private String replacePlaceholder(String sql, Object propertyValue) {
String value;
if (Objects.nonNull(propertyValue)) {
if (propertyValue instanceof String) {
value = "'" + propertyValue + "'";
} else if (propertyValue instanceof Date) {
value = "'" + DATE_TIME_FORMATTER
.format(((Date) propertyValue).toInstant().atZone(ZoneId.systemDefault()).toLocalDateTime())
+ "'";
} else if (propertyValue instanceof LocalDate) {
value = "'" + DATE_FORMATTER.format((LocalDate) propertyValue) + "'";
} else if (propertyValue instanceof LocalDateTime) {
value = "'" + DATE_TIME_FORMATTER.format((LocalDateTime) propertyValue) + "'";
} else {
value = propertyValue.toString();
}
} else {
value = "null";
}
return sql.replaceFirst("\\?", value);
}

/**
* 根据不同的超时时间打印不同颜色的字体,若超时时间大于默认的超时时间,打印红色字体,否则打印黄色字体
*
* @param str Str
* @param timeOut 超时时间
* @author liekkas 2020/12/08 10:50
*/
private void printColorString(String str, Long timeOut) {

if (timeOut < Constant.DEFAULT_TIME_OUT) {
LOGGER.info("-----------------------------------------------------------------------");

LOGGER.info("\033[33;4m" + str + "\033[0m");
LOGGER.info("-----------------------------------------------------------------------");
} else {
LOGGER.error("-----------------------------------------------------------------------");
LOGGER.error("\033[31;4m" + str + "\033[0m");
LOGGER.error("-----------------------------------------------------------------------");
}
}

/**
* <p>
* 内部的常量类,仅供本类使用
* </p>
*
* @author liekkas 2020/12/08 10:52
*/
private static class Constant {
public static final Long DEFAULT_TIME_OUT = 5000L;
}
}

 

配置方式有以下两种:

1:SpringBoot的注入配置方式

MybatisSqlConfig.java

import lombok.extern.slf4j.Slf4j;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Slf4j
@Configuration
public class MybatisSqlConfig {
@Bean
public SqlCostPlugins myPlugin() {
return new SqlCostPlugins();
}
}

2:SSM的配置方式

<?*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="SqlCostPlugins的全限定类名">
<property name="dialect" value="oracle"/>
</plugin>
</plugins>
</configuration>

mybatis自定义打印执行时间并格式化sql插件的更多相关文章

  1. mybatis 自定义插件的使用

    今天看了别人的mybatis的教学视频,自己手写了一个简单的自定义的插件,有些细节记录一下. 先看下mybatis的插件的一些说明: MyBatis 允许你在已映射语句执行过程中的某一点进行拦截调用. ...

  2. Mybatis框架(9)---Mybatis自定义插件生成雪花ID做为表主键项目

    Mybatis自定义插件生成雪花ID做为主键项目 先附上项目项目GitHub地址 spring-boot-mybatis-interceptor 有关Mybatis雪花ID主键插件前面写了两篇博客作为 ...

  3. mybatis日志,打印sql语句,输出sql

    mybatis日志,打印sql语句,输出sql<?xml version="1.0" encoding="UTF-8" ?><!DOCTYPE ...

  4. mybatis配置打印sql

    mybatis配置打印sql: <settings> <setting name="logImpl" value="STDOUT_LOGGING&quo ...

  5. Springboot中mybatis控制台打印sql语句

    Springboot中mybatis控制台打印sql语句 https://www.jianshu.com/p/3cfe5f6e9174 https://www.jianshu.com/go-wild? ...

  6. mybatis 控制台打印出来的sql 执行结果为空 但是将sql放到mysql执行有数据

    mybatis中的sql如下 select airln_Cd airlnCd,city_coordinate_j cityCoordinateJ,city_coordinate_w cityCoord ...

  7. SpringBoot使用Mybatis注解开发教程-分页-动态sql

    代码示例可以参考个人GitHub项目kingboy-springboot-data 一.环境配置 1.引入mybatis依赖 compile( //SpringMVC 'org.springframe ...

  8. 【mybatis源码学习】利用maven插件自动生成mybatis代码

    [一]在要生成代码的项目模块的pom.xml文件中添加maven插件 <!--mybatis代码生成器--> <plugin> <groupId>org.mybat ...

  9. Elasticsearch使用系列-基本查询和聚合查询+sql插件

    Elasticsearch使用系列-ES简介和环境搭建 Elasticsearch使用系列-ES增删查改基本操作+ik分词 Elasticsearch使用系列-基本查询和聚合查询+sql插件 Elas ...

随机推荐

  1. 狂神说Elasticsearch7.X学习笔记整理

    Elasticsearch概述 一.什么是Elasticsearch? Lucene简介 Lucene是一套用于全文检索和搜寻的开源程序库,由Apache软件基金会支持和提供 Lucene提供了一个简 ...

  2. Spring MVC工作原理及源码解析(四) ViewResolver实现原理及源码解析

    0.ViewResolver原理介绍 根据视图的名称将其解析为 View 类型的视图,如通过 ModelAndView 中的视图名称将其解析成 View,View 是用来渲染页面的,也就是将 Mode ...

  3. 本地Markdown上传图片

    本地Markdown上传图片 1.上传本地markdown文件到博客园 使用工具pycnblog 下载:https://github.com/dongfanger/PyCnblog 查看READ ME ...

  4. C++ primer plus读书笔记——第4章 复合类型

    第4章 复合类型 1. 如果将sizeof运算符用于数组名,得到的将是整个数组中的字节数. 2. 如果对数组的一部分进行初始化,则编译器把其他元素设置为0.因此,将数组中的所有元素初始化为0,只要显式 ...

  5. 2020BUAA-个人博客-案例分析

    个人博客作业-软件案例分析 项目 内容 北航2020软工 班级博客 作业要求 具体要求 我的课程目标 通过案例分析提升自己对于软件工程的认识 课程收获 分析软件,了解软件的定位 第一部分 调研,评测( ...

  6. [Java] 数据分析 -- 回归分析

    线性回归 需求:从文件读取数据对,计算回归函数及系数 实现1:commons.math的SimpleRegression,定义函数getData从文件读取数据返回SimpleRegression类 1 ...

  7. [bug] HDFS:hdfs missing blocks. The following files may be corrupted

    原因 HDFS数据块丢失,需要删除丢失块的元信息 bin/hadoop fsck / -delete 参考 https://blog.csdn.net/lixgjob/article/details/ ...

  8. [bug] Navicat 连 虚拟机MySQL

    参考 https://www.cnblogs.com/brankoliu/p/10845491.html https://blog.csdn.net/qq_40087740/article/detai ...

  9. 一文搞懂spring的常用注解

    spring传统做法是使用xml文件对bean进行注入和配置.通过使用spring提供的注解,可以极大的降低配置xml文件的繁琐.本文将介绍常用的注解. 一@Autowired Autowired意为 ...

  10. Python 库整理【收藏】

    库名称简介 Chardet字符编码探测器,可以自动检测文本.网页.xml的编码. colorama主要用来给文本添加各种颜色,并且非常简单易用. Prettytable主要用于在终端或浏览器端构建格式 ...