MyBatis插件开发-实现SQL执行耗时监控

MyBatis插件开发实现SQL执行耗时监控

MyBatis插件通过拦截器机制实现SQL执行耗时监控,核心是拦截StatementHandlerExecutor相关方法,记录执行前后时间差。

创建自定义拦截器类

自定义拦截器需实现Interceptor接口,添加@Intercepts注解定义拦截目标。以下示例拦截StatementHandlerqueryupdate方法:

java 复制代码
@Intercepts({
    @Signature(type = StatementHandler.class, method = "query", 
        args = {Statement.class, ResultHandler.class}),
    @Signature(type = StatementHandler.class, method = "update", 
        args = {Statement.class})
})
public class SqlCostInterceptor implements Interceptor {
    @Override
    public Object intercept(Invocation invocation) throws Throwable {
        long startTime = System.currentTimeMillis();
        Object result = invocation.proceed();
        long endTime = System.currentTimeMillis();
        System.out.println("SQL执行耗时: " + (endTime - startTime) + "ms");
        return result;
    }
}

注册拦截器到MyBatis配置

在MyBatis配置文件中添加拦截器,或通过编程方式注册:

XML 复制代码
<plugins>
    <plugin interceptor="com.example.SqlCostInterceptor"/>
</plugins>

或通过SqlSessionFactory配置:

java 复制代码
@Bean
public SqlSessionFactory sqlSessionFactory(DataSource dataSource) throws Exception {
    SqlSessionFactoryBean factory = new SqlSessionFactoryBean();
    factory.setDataSource(dataSource);
    factory.setPlugins(new SqlCostInterceptor());
    return factory.getObject();
}

增强监控信息输出

可通过MappedStatement获取SQL详细信息,完善监控日志:

java 复制代码
StatementHandler handler = (StatementHandler) invocation.getTarget();
BoundSql boundSql = handler.getBoundSql();
MappedStatement ms = (MappedStatement) 
    ReflectUtil.getFieldValue(handler, "mappedStatement");

String sqlId = ms.getId();
String sql = boundSql.getSql();

使用SLF4J记录日志

推荐整合日志框架替代System.out

java 复制代码
private static final Logger logger = LoggerFactory.getLogger(SqlCostInterceptor.class);

// 在intercept方法中记录
logger.debug("SQL [{}]执行耗时: {}ms", sqlId, (endTime - startTime));

注意事项

拦截器顺序可能影响执行结果,需通过@Order注解或配置顺序控制。避免在拦截器中执行耗时操作,防止性能影响。生产环境建议添加阈值报警机制,对慢SQL进行标记。

相关推荐
lly202406几秒前
C 标准库 - `<stdio.h>`
开发语言
沫璃染墨3 分钟前
C++ string 从入门到精通:构造、迭代器、容量接口全解析
c语言·开发语言·c++
jwn9993 分钟前
Laravel6.x核心特性全解析
开发语言·php·laravel
迷藏4945 分钟前
**发散创新:基于Rust实现的开源合规权限管理框架设计与实践**在现代软件架构中,**权限控制(RBAC)** 已成为保障
java·开发语言·python·rust·开源
功德+n34 分钟前
Linux下安装与配置Docker完整详细步骤
linux·运维·服务器·开发语言·docker·centos
明日清晨36 分钟前
python扫码登录dy
开发语言·python
我是唐青枫42 分钟前
C#.NET gRPC 深入解析:Proto 定义、流式调用与服务间通信取舍
开发语言·c#·.net
JJay.1 小时前
Kotlin 高阶函数学习指南
android·开发语言·kotlin
bazhange1 小时前
python如何像matlab一样使用向量化替代for循环
开发语言·python·matlab
jinanwuhuaguo1 小时前
截止到4月8日,OpenClaw 2026年4月更新深度解读剖析:从“能力回归”到“信任内建”的范式跃迁
android·开发语言·人工智能·深度学习·kotlin