mybatis拦截器打印完整带参数sql,可直接放入数据库工具执行

本文主要是介绍mybatis拦截器打印完整带参数sql,可直接放入数据库工具执行,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

如果你使用的是mybatisplus,那至少有3种方法打印sql, 如下博客

mybatisplus开启sql打印的三种方式_mybatisplus打印sql语句配置-CSDN博客

但是其中2种方法参数会打印问号,不能直接放入数据库工具执行

而使用p6spy插件需要引入新依赖,有未知的性能消耗。

因此,这里分享一种通过mybatis拦截器,实现sql打印

代码

package com.gree;import org.apache.ibatis.cache.CacheKey;
import org.apache.ibatis.executor.Executor;
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.session.RowBounds;
import org.apache.ibatis.type.TypeHandlerRegistry;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Component;import java.sql.Timestamp;
import java.text.SimpleDateFormat;
import java.util.*;/*** 基于Mybatis Plus的SQL输出拦截器。* 完美的输出打印 SQL 及执行时长、statement。* 注意:该插件有性能损耗,不建议生产环境使用。*/@Intercepts(value = {@Signature(type = Executor.class, method = "update", args = {MappedStatement.class, Object.class}),@Signature(type = Executor.class, method = "query", args = {MappedStatement.class, Object.class, RowBounds.class, ResultHandler.class, CacheKey.class, BoundSql.class}),
//        @Signature(type = Executor.class, method = "query", args = {MappedStatement.class, Object.class, RowBounds.class, ResultHandler.class})
})
//@Component用于注入到spring中,后续会被mybatis-plus-boot-starter中的代码从spring中取出并注入到mybatis中
@Component
public class MybatisPlusPrintSqlInterceptor implements Interceptor {Logger logger = LoggerFactory.getLogger(MybatisPlusPrintSqlInterceptor.class);@Overridepublic Object intercept(Invocation invocation) throws Throwable {//1. 执行sqllong proceedStart = System.currentTimeMillis();Object returnValue = null;Exception proceedException = null;//执行sql时catch下异常,即使sql语法报错,也要打印完整sqltry {returnValue = invocation.proceed();} catch (Exception e) {proceedException = e;}long proceedCost = System.currentTimeMillis() - proceedStart;//2.打印sqllong printBegin = System.currentTimeMillis();MappedStatement mappedStatement = (MappedStatement) invocation.getArgs()[0];//部分mybatisplus拦截器内部可能会对Args中的sql进行修改,因此从Args中获取boundSql更接近与真实执行sqlBoundSql boundSql = null;for (int i = invocation.getArgs().length - 1; i >= 0; i--) {if (invocation.getArgs()[i] instanceof BoundSql) {boundSql = (BoundSql) invocation.getArgs()[i];}}if (boundSql == null) {Object parameter = null;if (invocation.getArgs().length > 1) {parameter = invocation.getArgs()[1];}boundSql = mappedStatement.getBoundSql(parameter);}String statement = mappedStatement.getId();Configuration configuration = mappedStatement.getConfiguration();showSql(configuration, boundSql, proceedCost, statement);long printEnd = System.currentTimeMillis();System.out.println("本次打印sql消耗时间:" + (printEnd - printBegin));//3. sql执行异常的报错扔出去if (proceedException != null) {throw proceedException;}return returnValue;}private void showSql(Configuration configuration, BoundSql boundSql, long elapsed, String statement) {String logText = formatMessage(elapsed, getSqlWithValues(boundSql.getSql(), buildParameterValues(configuration, boundSql)), statement);if (Boolean.TRUE == false) {// 打印红色 SQL 日志System.err.println(logText);} else {logger.info("\n{}", logText);}}// com.baomidou.mybatisplus.core.MybatisParameterHandler#setParametersprivate static Map<Integer, Object> buildParameterValues(Configuration configuration, BoundSql boundSql) {Object parameterObject = boundSql.getParameterObject();// ParameterMapping描述参数,包括属性、名称、表达式、javaType、jdbcType、typeHandler等信息List<ParameterMapping> parameterMappings = boundSql.getParameterMappings();if (parameterMappings != null) {Map<Integer, Object> parameterValues = new HashMap<>();//类型处理器用于注册所有的 TypeHandler,并建立 Jdbc 类型、JDBC 类型与 TypeHandler 之间的对应关系TypeHandlerRegistry typeHandlerRegistry = configuration.getTypeHandlerRegistry();for (int i = 0; i < parameterMappings.size(); i++) {ParameterMapping parameterMapping = parameterMappings.get(i);if (parameterMapping.getMode() != ParameterMode.OUT) {Object value;String propertyName = parameterMapping.getProperty();if (boundSql.hasAdditionalParameter(propertyName)) { // issue #448 ask first for additional paramsvalue = 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);}parameterValues.put(i, new Value(value));}}return parameterValues;}return Collections.emptyMap();}public static String formatMessage( long elapsed, String sql, String statement) {return (sql != null && sql != "") ?
//                " Consume Time:" + elapsed + " ms "  + " (" + statement + ")" + "  Execute SQL:" + sql.replaceAll("[\\s]+", " ")//" Consume Time:" + elapsed + " ms ,  Execute SQL:" + sql.replaceAll("[\\s]+", " ")
//                " Consume Time:" + elapsed + " ms ,  Execute SQL:" + sql" 本次执行sql耗时:" + elapsed + " ms "  + " (" + statement + ")" + "  执行的sql打印:" + sql: "";}@Overridepublic Object plugin(Object target) {return Plugin.wrap(target, this);}@Overridepublic void setProperties(Properties properties0) {}public static String getSqlWithValues(String statementQuery, Map<Integer, Object> parameterValues) {final StringBuilder sb = new StringBuilder();// iterate over the characters in the query replacing the parameter placeholders// with the actual valuesint currentParameter = 0;for (int pos = 0; pos < statementQuery.length(); pos++) {char character = statementQuery.charAt(pos);if (statementQuery.charAt(pos) == '?' && currentParameter <= parameterValues.size()) {// replace with parameter valueObject value = parameterValues.get(currentParameter);sb.append(value != null ? value.toString() : new Value().toString());currentParameter++;} else {sb.append(character);}}return sb.toString();}/*** 基于p6spy的简易数据类型转换类。** @author laiqi* @date 2023-4-4*/public static class Value {public static final String NORM_DATETIME_PATTERN = "yyyy-MM-dd HH:mm:ss";public static final String databaseDialectDateFormat = NORM_DATETIME_PATTERN;public static final String databaseDialectTimestampFormat = NORM_DATETIME_PATTERN;public static final String databaseDialectBooleanFormat = "numeric";private Object value;public Value(Object valueToSet) {this();this.value = valueToSet;}public Value() {}public Object getValue() {return value;}public void setValue(Object value) {this.value = value;}@Overridepublic String toString() {return convertToString(this.value);}public String convertToString(Object value) {String result;if (value == null) {result = "NULL";} else {if (value instanceof byte[]) {result = new String((byte[]) value);} else if (value instanceof Timestamp) {result = new SimpleDateFormat(databaseDialectTimestampFormat).format(value);} else if (value instanceof Date) {result = new SimpleDateFormat(databaseDialectDateFormat).format(value);} else if (value instanceof Boolean) {if ("numeric".equals(databaseDialectBooleanFormat)) {result = Boolean.FALSE.equals(value) ? "0" : "1";} else {result = value.toString();}} else {result = value.toString();}result = quoteIfNeeded(result, value);}return result;}private String quoteIfNeeded(String stringValue, Object obj) {if (stringValue == null) {return null;}if (Number.class.isAssignableFrom(obj.getClass()) || Boolean.class.isAssignableFrom(obj.getClass())) {return stringValue;} else {return "'" + escape(stringValue) + "'";}}private String escape(String stringValue) {return stringValue.replaceAll("'", "''");}}
}

注意

1. 如果你的项目里有mybatis-plus-boot-starter,那只需要把这个类复制到springboot能被扫描到的地方就行,如果项目中没有mybatis-plus-boot-starter,那你可能还需要手动将这个拦截器注入到mybatis里

2. 如果你的项目有有mybatis-plus-boot-starter,并且存在其他mybatis拦截器或者mybatisplus的拦截器,那你需要调整bean的创建顺序,使得该打印sql的拦截器最早注入spring中,这样mybatisplus从spring中获取拦截器的时候,该打印sql的拦截器排在最前面,注入mybatis后,该打印sql的拦截器就会最后执行,你能获得更加准确的sql打印结果

其他参考博客

本文主要代码参考基于以下博客

基于Mybatis Plus的SQL输出拦截器。完美的输出打印 SQL 及执行时长、statement。_mybatissqlprintlnterceptorjava-CSDN博客

这篇关于mybatis拦截器打印完整带参数sql,可直接放入数据库工具执行的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1106264

相关文章

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

mybatis的整体架构

mybatis的整体架构分为三层: 1.基础支持层 该层包括:数据源模块、事务管理模块、缓存模块、Binding模块、反射模块、类型转换模块、日志模块、资源加载模块、解析器模块 2.核心处理层 该层包括:配置解析、参数映射、SQL解析、SQL执行、结果集映射、插件 3.接口层 该层包括:SqlSession 基础支持层 该层保护mybatis的基础模块,它们为核心处理层提供了良好的支撑。

大模型研发全揭秘:客服工单数据标注的完整攻略

在人工智能(AI)领域,数据标注是模型训练过程中至关重要的一步。无论你是新手还是有经验的从业者,掌握数据标注的技术细节和常见问题的解决方案都能为你的AI项目增添不少价值。在电信运营商的客服系统中,工单数据是客户问题和解决方案的重要记录。通过对这些工单数据进行有效标注,不仅能够帮助提升客服自动化系统的智能化水平,还能优化客户服务流程,提高客户满意度。本文将详细介绍如何在电信运营商客服工单的背景下进行

SQL中的外键约束

外键约束用于表示两张表中的指标连接关系。外键约束的作用主要有以下三点: 1.确保子表中的某个字段(外键)只能引用父表中的有效记录2.主表中的列被删除时,子表中的关联列也会被删除3.主表中的列更新时,子表中的关联元素也会被更新 子表中的元素指向主表 以下是一个外键约束的实例展示

基于MySQL Binlog的Elasticsearch数据同步实践

一、为什么要做 随着马蜂窝的逐渐发展,我们的业务数据越来越多,单纯使用 MySQL 已经不能满足我们的数据查询需求,例如对于商品、订单等数据的多维度检索。 使用 Elasticsearch 存储业务数据可以很好的解决我们业务中的搜索需求。而数据进行异构存储后,随之而来的就是数据同步的问题。 二、现有方法及问题 对于数据同步,我们目前的解决方案是建立数据中间表。把需要检索的业务数据,统一放到一张M

如何去写一手好SQL

MySQL性能 最大数据量 抛开数据量和并发数,谈性能都是耍流氓。MySQL没有限制单表最大记录数,它取决于操作系统对文件大小的限制。 《阿里巴巴Java开发手册》提出单表行数超过500万行或者单表容量超过2GB,才推荐分库分表。性能由综合因素决定,抛开业务复杂度,影响程度依次是硬件配置、MySQL配置、数据表设计、索引优化。500万这个值仅供参考,并非铁律。 博主曾经操作过超过4亿行数据

性能分析之MySQL索引实战案例

文章目录 一、前言二、准备三、MySQL索引优化四、MySQL 索引知识回顾五、总结 一、前言 在上一讲性能工具之 JProfiler 简单登录案例分析实战中已经发现SQL没有建立索引问题,本文将一起从代码层去分析为什么没有建立索引? 开源ERP项目地址:https://gitee.com/jishenghua/JSH_ERP 二、准备 打开IDEA找到登录请求资源路径位置

MySQL数据库宕机,启动不起来,教你一招搞定!

作者介绍:老苏,10余年DBA工作运维经验,擅长Oracle、MySQL、PG、Mongodb数据库运维(如安装迁移,性能优化、故障应急处理等)公众号:老苏畅谈运维欢迎关注本人公众号,更多精彩与您分享。 MySQL数据库宕机,数据页损坏问题,启动不起来,该如何排查和解决,本文将为你说明具体的排查过程。 查看MySQL error日志 查看 MySQL error日志,排查哪个表(表空间

Andrej Karpathy最新采访:认知核心模型10亿参数就够了,AI会打破教育不公的僵局

夕小瑶科技说 原创  作者 | 海野 AI圈子的红人,AI大神Andrej Karpathy,曾是OpenAI联合创始人之一,特斯拉AI总监。上一次的动态是官宣创办一家名为 Eureka Labs 的人工智能+教育公司 ,宣布将长期致力于AI原生教育。 近日,Andrej Karpathy接受了No Priors(投资博客)的采访,与硅谷知名投资人 Sara Guo 和 Elad G

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�