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进行标记。

相关推荐
考虑考虑16 小时前
Mybatis实现批量插入
java·后端·mybatis
咖啡八杯17 小时前
GoF设计模式——中介者模式
java·后端·spring·设计模式
青石路20 小时前
记一次多JDK版本问题的排查,一坑套一坑,差点没爬上来
java
像我这样帅的人丶你还1 天前
Java 后端详解(五):Redis 缓存
java·后端·全栈
plainGeekDev1 天前
GreenDAO → Room
android·java·kotlin
亦暖筑序1 天前
Java 8老系统AI Workflow实战:把一次性AI对话升级成可恢复工作流
java·后端
敲代码的彭于晏1 天前
Bean 生命周期完全图解:前端同学也能看懂的 Spring 核心机制
java·前端·后端
plainGeekDev1 天前
ButterKnife → ViewBinding
android·java·kotlin
像我这样帅的人丶你还2 天前
Java 后端详解(四):分页与搜索
java·javascript·后端
她的男孩2 天前
数据权限为什么不能只靠注解?Forge 的 Mapper 层 SQL 改写源码拆解
java·后端·架构