MyBatis-Plus输出完整SQL(带参数)的三种方案

 更新时间:2025年02月06日 10:02:03   作者:( •̀∀•́ )920  
当我们使用 mybatis-plus 时,可能会遇到SQL 不能直接执行,调试也不方便的情况,那么,如何打印完整 SQL(带参数)呢?本篇文章将介绍 3 种实现方式,并对比它们的优缺点,需要的朋友可以参考下

为什么 MyBatis Plus 默认 SQL 日志没有参数?

当你使用 mybatis-plus 时,可能会遇到这样的情况:

Preparing: SELECT id, name, `desc` FROM author WHERE name = ?
Parameters: 刘禹锡(String)

这导致 SQL 不能直接执行,调试也不方便。那么,如何打印完整 SQL(带参数)呢?本篇文章将介绍 3 种实现方式,并对比它们的优缺点。

方案 1:使用 SqlLogInterceptor(推荐)

从 MyBatis Plus 3.5.3 版本 开始,官方提供了 SqlLogInterceptor,可以自动替换 SQL 语句中的 ? 为实际参数。

配置方式:

import com.baomidou.mybatisplus.extension.plugins.MybatisPlusInterceptor;
import com.baomidou.mybatisplus.extension.plugins.inner.SqlLogInterceptor;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
public class MyBatisPlusConfig {

    @Bean
    public MybatisPlusInterceptor mybatisPlusInterceptor() {
        MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
        interceptor.addInnerInterceptor(new SqlLogInterceptor()); // ✅ 让 SQL 带参数
        return interceptor;
    }
}

效果:

Executing SQL: SELECT id, name, `desc` FROM author WHERE name = '刘禹锡'

优点:

  • 官方支持,无侵入,简单易用
  • 低性能开销,仅影响日志输出
  • 适用于 MyBatis Plus

缺点:

  • 只适用于 MyBatis Plus(普通 MyBatis 需要用方案 2)

方案 2:自定义 MyBatis Interceptor

如果你使用的是 原生 MyBatis,或者想要更灵活的日志格式,可以 自定义 Interceptor

编写 Interceptor:

@Intercepts({
    @Signature(type = Executor.class, method = "query", args = {
        MappedStatement.class, Object.class, RowBounds.class, ResultHandler.class}),
    @Signature(type = Executor.class, method = "update", args = {
        MappedStatement.class, Object.class})
})
@Slf4j
public class SqlInterceptor implements Interceptor {
    @Override
    public Object intercept(Invocation invocation) throws Throwable {
        long startTime = System.currentTimeMillis();
        Object proceed = invocation.proceed();
        long endTime = System.currentTimeMillis();

        String sql = generateSql(invocation);
        log.info("\n 执行SQL耗时:{}ms \n 执行SQL:{}", endTime - startTime, sql);
        return proceed;
    }

    private static String generateSql(Invocation invocation) {
        MappedStatement statement = (MappedStatement) invocation.getArgs()[0];
        Object parameter = invocation.getArgs().length > 1 ? invocation.getArgs()[1] : null;
        BoundSql boundSql = statement.getBoundSql(parameter);

        String sql = boundSql.getSql().replaceAll("[\\s]+", " ");
        for (ParameterMapping param : boundSql.getParameterMappings()) {
            Object value = boundSql.getAdditionalParameter(param.getProperty());
            sql = sql.replaceFirst("\\?", Matcher.quoteReplacement(getParameterValue(value)));
        }
        return sql;
    }

    private static String getParameterValue(Object object) {
        return object instanceof String ? "'" + object + "'" : String.valueOf(object);
    }

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

优点:

  • 适用于 原生 MyBatis 和 MyBatis Plus
  • 可自定义日志格式,控制输出内容

缺点:

  • 需要手动编写,稍微复杂一点
  • 性能开销略大(需要解析 SQL 并替换参数)

方案 3:开启 MyBatis 日志(手动拼接 SQL)

如果只是 临时调试,可以在 application.yml 配置 MyBatis 日志:

mybatis-plus:
  configuration:
    log-impl: org.apache.ibatis.logging.stdout.StdOutImpl

然后,手动拼接 SQL 进行调试。

优点:

  • 无代码改动,适合临时调试

缺点:

  • 仍然是 Preparing: xxx 和 Parameters: xxx 的分离格式
  • 不能直接复制执行

最佳方案对比与选择

方案适用场景侵入性性能开销适用框架
SqlLogInterceptorMyBatis PlusMyBatis Plus
自定义 Interceptor需要特殊格式中等MyBatis & MyBatis Plus
MyBatis 日志临时调试MyBatis & MyBatis Plus

结论:哪种方式适合你?

  • 推荐:使用 SqlLogInterceptor,简单、无侵入,适用于 MyBatis Plus
  • 进阶:自定义 Interceptor,适用于 MyBatis,可自定义日志格式。
  • 临时调试:开启 MyBatis 日志,但不自动替换 ?

到此这篇关于MyBatis-Plus输出完整SQL(带参数)的三种方案的文章就介绍到这了,更多相关MyBatis-Plus输出完整SQL内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java覆盖第三方jar包中的某一个类的实现方法

    Java覆盖第三方jar包中的某一个类的实现方法

    在我们日常的开发中,经常需要使用第三方的 jar 包,有时候我们会发现第三方的 jar 包中的某一个类有问题,或者我们需要定制化修改其中的逻辑,那么应该如何实现呢,本文给大家介绍了Java覆盖第三方jar包中的某一个类的实现方法,需要的朋友可以参考下
    2025-02-02
  • Mybatis plus逻辑删除失败的BUG操作

    Mybatis plus逻辑删除失败的BUG操作

    这篇文章主要介绍了Mybatis plus逻辑删除失败的BUG操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-11-11
  • Java基于动态规划法实现求最长公共子序列及最长公共子字符串示例

    Java基于动态规划法实现求最长公共子序列及最长公共子字符串示例

    这篇文章主要介绍了Java基于动态规划法实现求最长公共子序列及最长公共子字符串,简单描述了动态规划法的概念、原理,并结合实例形式分析了Java使用动态规划法求最长公共子序列以及最长公共子字符串相关实现技巧,需要的朋友可以参考下
    2018-08-08
  • Hadoop源码分析五hdfs架构原理剖析

    Hadoop源码分析五hdfs架构原理剖析

    本篇是Hadoop源码分析系列文章第五篇,主要介绍Hadoop的hdfs架构原理剖析,后续本系列文章会持续更新,有需要的朋友可以借鉴参考下
    2021-09-09
  • 分享几个提高Java性能的高效用法

    分享几个提高Java性能的高效用法

    这篇文章主要介绍了分享几个提高Java性能的高效用法 ,需要的朋友可以参考下
    2014-10-10
  • springboot项目同时启动web服务和grpc服务的方法

    springboot项目同时启动web服务和grpc服务的方法

    本文主要介绍了springboot项目同时启动web服务和grpc服务的方法,通过实际代码示例展示了实现,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2024-02-02
  • 基于自定义BufferedReader中的read和readLine方法

    基于自定义BufferedReader中的read和readLine方法

    下面小编就为大家分享一篇基于自定义BufferedReader中的read和readLine方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2017-12-12
  • 详解Java并发编程之内置锁(synchronized)

    详解Java并发编程之内置锁(synchronized)

    这篇文章主要介绍了Java并发编程之内置锁(synchronized)的相关知识,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-03-03
  • SpringMVC表单提交参数400错误解决方案

    SpringMVC表单提交参数400错误解决方案

    这篇文章主要介绍了SpringMVC表单提交参数400错误解决方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-10-10
  • SpringBoot实现文件断点续传功能详解

    SpringBoot实现文件断点续传功能详解

    在处理大文件传输或网络不稳定的情况下,文件断点续传功能显得尤为重要,本文将详细介绍如何使用Spring Boot实现文件的断点续传功能,需要的可以了解下
    2025-04-04

最新评论