Springboot中分析SQL性能的两种方式详解

2025-02-21 17:50

本文主要是介绍Springboot中分析SQL性能的两种方式详解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

《Springboot中分析SQL性能的两种方式详解》文章介绍了SQL性能分析的两种方式:MyBatis-Plus性能分析插件和p6spy框架,MyBatis-Plus插件配置简单,适用于开发和测试环...

SQL性能分析的两种方式:

功能介绍

  • 记录 SQL 执行时间,超过阈值会进行警告
  • 打印完整的 SQL 语句,便于调试和优化
  • 适用于开发和测试环境,生产环境建议关闭

实现方式:

方式一:使用 MyBATis-Plus 性能分析插件

首先需要在MyBatis-Plus配置类中配置PerformanceInterceptor插件:

@Configuration
public class MyBatisPlusConfig {
    @Bean
    public MybatisPlusInterceptor mybatisPlusInterceptor() {
        MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
        // 添加分页拦截器(如果使用分页)
        interceptor.addInnerInterceptor(new PagiphpnationInnerInterceptor(DbType.mysql));
        // 添加性能分析拦截器
        PerformanceInterceptor performanceInterceptor = new PerformanceInterceptor();
        performanceInterceptor.setMaxTandroidime(500); // SQL 最大执行时间(毫秒),超过自动警告
        performanceInterceptor.setFormat(true); // 是否格式化 SQL 语句
        interceptor.addInnerInterceptor(performanceInterceptor);
        return interceptor;
    }
}

配置项说明:

配置项说明示例
setMaxTime(long maxTime)设置 SQL 执行的最大时长(单位:ms),超过时间将抛出异常,默认 -1(不限制)performanceInterceptor.setMaxTime(500); // 超过 500ms 警告
setFormat(boolean format)是否格式化 SQL,默认 false,开启后会美化 SQL 输出performanceInterceptor.setFormat(true);

日志输出:

当查询执行时间超过 maxTime 限制时,控制台会输出:

[SQL] ==> Preparing: SELECT * FROM orders WHERE status = ?
[SQL] ==> Parameters: 1(Integer)
[SQL] <== Total Time: 850 ms (超过最大时间 500ms,可能存在性能问题)

由于MyBatis-Plus 3.4.0 以后已经将上述插件弃用,所以如果你使用高版本的MyBatis-Plus,推荐使用方式二.

方式二:使用p6spy框架 效果图:

Springboot中分析SQL性能的两种方式详解

实现步骤:

1.引入pom依赖:

<dependency>
    <groupId>p6spy</groupId>
    <artifactId>p6spy</artifactId>
    <version>3.9.1</version>
</dependency>

2.首先在 springboot配置类中加入配置:

  • 把原来的JDBC Driver替换为com.p6spy.engine.spy.P6SpyDriver.
  • 在原来urljdbc:后面添加p6spy:,比如jdbc:p6spy:mysql://127.0.0.1:3306/databse
  • 添加p6spy的配置文件spy.properties.

配置文件内容如下:

module.log=com.p6spy.engine.logging.P6LogFactory,com.p6spy.engine.outage.P6OutageFactory  
# 自定义日志打印  
logMessageFormat=warren.reggie.common.P6SpyLogger  
#logMessageFormat=com.p6spy.engine.spy.appender.CustomLineFormat  
#customLogMessageFormat=%(currentTime) | SQL耗时: %(executionTime) ms | 连接信息: %(category)-%(connectionId) | 执行语句: %(sql)# 使用控制台记录sql  
appender=com.p6spy.engine.spy.appender.StdoutLogger  
## 配置记录Log例外  
excludecategories=info,debug,result,batc,resultset  
# 设置使用p6spy driver来做代理  
deregisterdrivers=true  
# 日期格式  
dateformat=yyyy-MM-dd HH:mm:ss  
# 实际驱动  
driverlist=com.mysql.jdbc.Driver  
# 是否开启慢SQL记录  
outagedetection=true  
# 慢SQL记录标准秒  
outagedetectioninterval=2

3.自定义日志输出格式:

首先创建P6SpyLogger类:

package warren.reggie.common;  
import com.p6spy.engine.spy.appender.MessageFormattingStrategy;  
import Java.time.LocalDateTime;  
import java.time.format.DateTimeFormatter;  
/**  
 * 自定义 P6Spy SQL 日志格式(带颜色)  
 * author: Warren  
 */public class P6SpyLogger implements MessageFormattingStrategy {  
    // ANSI 颜色代码  
    private static final String RESET = "\u001B[0m";  // 重置颜色  
    private static final String RED = "\u001B[31m";   // 红色(高亮错误)  
    private static final String GREEN = "\u001B[32m"; // 绿色(SQL 语句)  
    private static final String YELLOW = "\u001B[33m";// 黄色(执行时间)  
    private static final String BLUE = "\u001B[34m";  // 蓝色(分类)  
    private static final String CYAN = "\u001B[36m";  // 青色(连接 ID)  
    /**  
     * 自定义 SQL 日志格式(带颜色)  
     *  
     * @param connectionId 连接 ID编程  
     * @param now 当前时间(P6Spy 传递的)  
     * @param elapsed SQL 执行时间(ms)  
     * @param category SQL 类型(如 statement、commit、rollback)  
     * @param prepared 预编译 SQL(带 ? 占位符)  
     * @param sql 真实 SQL 语句(占位符替换后的)  
     * @param url 数据库连接 URL  
     * @return 格式化后的日志字符串  
     */  
    @Override  
    public String formatMessage(int connectionId, String now, long elapsed, String category, String prepared, String sql, String url) {  
        // 过滤空 http://www.chinasem.cnSQL        if (sql == null || sql.trim().isEmpty()) {  
            return "";  
        }  
        // 格式化当前时间  
        String currentTime = LocalDateTime.now().format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"));  
        // 构造带颜色的日志输出  
        return String.format(  
                "%s[%s]%s | %s耗时: %d ms%s | %s连接: %d%s | %s分类: %s%s\n%s执行 SQL: %s%s;\n",  
                CYAN, currentTime, RESET,           // 时间(青色)  
                YELLOW, elapsed, RESET,            // 执行时间(黄色)  
                BLUE, connectionId, RESET,         // 连接 ID(蓝色)  
                RED, category, RESET,              // 分类(红色)  
                GREEN, sql.trim(), RESET           // SQL 语句(绿色)  
        );  
    }  
}

然后将配置文件中的格式化器属性改为自己的类 :

# 自定义日志打印  
logMessageFormat=warren.reggie.common.P6SpyLogger  

到此这篇关于Spri编程ngboot中分析SQL性能的两种方式详解的文章就介绍到这了,更多相关Springboot 分析sql性能内容请搜索China编程(www.chinasem.cn)以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程China编程(www.chinasem.cn)!

这篇关于Springboot中分析SQL性能的两种方式详解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Docker镜像pull失败两种解决办法小结

《Docker镜像pull失败两种解决办法小结》有时候我们在拉取Docker镜像的过程中会遇到一些问题,:本文主要介绍Docker镜像pull失败两种解决办法的相关资料,文中通过代码介绍的非常详细... 目录docker 镜像 pull 失败解决办法1DrQwWCocker 镜像 pull 失败解决方法2总

SpringBoot启动报错的11个高频问题排查与解决终极指南

《SpringBoot启动报错的11个高频问题排查与解决终极指南》这篇文章主要为大家详细介绍了SpringBoot启动报错的11个高频问题的排查与解决,文中的示例代码讲解详细,感兴趣的小伙伴可以了解一... 目录1. 依赖冲突:NoSuchMethodError 的终极解法2. Bean注入失败:No qu

Java异常架构Exception(异常)详解

《Java异常架构Exception(异常)详解》:本文主要介绍Java异常架构Exception(异常),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1. Exception 类的概述Exception的分类2. 受检异常(Checked Exception)

使用Java实现通用树形结构构建工具类

《使用Java实现通用树形结构构建工具类》这篇文章主要为大家详细介绍了如何使用Java实现通用树形结构构建工具类,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录完整代码一、设计思想与核心功能二、核心实现原理1. 数据结构准备阶段2. 循环依赖检测算法3. 树形结构构建4. 搜索子

C#基础之委托详解(Delegate)

《C#基础之委托详解(Delegate)》:本文主要介绍C#基础之委托(Delegate),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1. 委托定义2. 委托实例化3. 多播委托(Multicast Delegates)4. 委托的用途事件处理回调函数LINQ

找不到Anaconda prompt终端的原因分析及解决方案

《找不到Anacondaprompt终端的原因分析及解决方案》因为anaconda还没有初始化,在安装anaconda的过程中,有一行是否要添加anaconda到菜单目录中,由于没有勾选,导致没有菜... 目录问题原因问http://www.chinasem.cn题解决安装了 Anaconda 却找不到 An

Spring定时任务只执行一次的原因分析与解决方案

《Spring定时任务只执行一次的原因分析与解决方案》在使用Spring的@Scheduled定时任务时,你是否遇到过任务只执行一次,后续不再触发的情况?这种情况可能由多种原因导致,如未启用调度、线程... 目录1. 问题背景2. Spring定时任务的基本用法3. 为什么定时任务只执行一次?3.1 未启用

浅谈mysql的sql_mode可能会限制你的查询

《浅谈mysql的sql_mode可能会限制你的查询》本文主要介绍了浅谈mysql的sql_mode可能会限制你的查询,这个问题主要说明的是,我们写的sql查询语句违背了聚合函数groupby的规则... 目录场景:问题描述原因分析:解决方案:第一种:修改后,只有当前生效,若是mysql服务重启,就会失效;

MySQL多列IN查询的实现

《MySQL多列IN查询的实现》多列IN查询是一种强大的筛选工具,它允许通过多字段组合快速过滤数据,本文主要介绍了MySQL多列IN查询的实现,具有一定的参考价值,感兴趣的可以了解一下... 目录一、基础语法:多列 IN 的两种写法1. 直接值列表2. 子查询二、对比传统 OR 的写法三、性能分析与优化1.

在C#中调用Python代码的两种实现方式

《在C#中调用Python代码的两种实现方式》:本文主要介绍在C#中调用Python代码的两种实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C#调用python代码的方式1. 使用 Python.NET2. 使用外部进程调用 Python 脚本总结C#调