分享

MyBatis 插件 : 打印 SQL 及其执行时间

 释皇天 2017-08-30

来源:五月的仓颉,



Plugins


摘一段来自MyBatis官方文档的文字。


MyBatis允许你在某一点拦截已映射语句执行的调用。默认情况下,MyBatis允许使用插件来拦截方法调用


  • Executor(update、query、flushStatements、commint、rollback、getTransaction、close、isClosed)

  • ParameterHandler(getParameterObject、setParameters)

  • ResultSetHandler(handleResultSets、handleOutputParameters)

  • StatementHandler(prepare、parameterize、batch、update、query)


这些类中方法的详情可以通过查看每个方法的签名来发现,而且它们的源代码存在于MyBatis发行包中。你应该理解你所覆盖方法的行为,假设你所做的要比监视调用要多。如果你尝试修改或覆盖一个给定的方法,你可能会打破MyBatis的核心。这是低层次的类和方法,要谨慎使用插件。


插件示例:打印每条SQL语句及其执行时间


以下通过代码来演示一下如何使用MyBatis的插件,要演示的场景是:打印每条真正执行的SQL语句及其执行的时间。这是一个非常有用的需求,MyBatis本身的日志可以记录SQL,但是有以下几个问题:


  1. MyBatis日志打印出来的SQL日志,参数都被占位符”?”替换,无法知道真正执行的SQL语句中的参数是什么

  2. MyBatis日志打印出来的SQL日志,有大量的换行符,通常一句SQL语句要通过十几行显示,阅读体验非常差

  3. 无法记录SQL执行时间,有SQL执行时间就可以精准定位到执行时间比较慢的SQL


写MyBatis插件非常简单,只需要实现Interceptor接口即可,我这里将我的Interceptor命名为SqlCostInterceptor:


/**

 * Sql执行时间记录拦截器 

 */

@Intercepts({@Signature(type = StatementHandler.class, method = 'query', args = {Statement.class, ResultHandler.class}),

    @Signature(type = StatementHandler.class, method = 'update', args = {Statement.class}),

    @Signature(type = StatementHandler.class, method = 'batch', args = { Statement.class })})

public class SqlCostInterceptor implements Interceptor {

 

    @Override

    public Object intercept(Invocation invocation) throws Throwable {

        Object target = invocation.getTarget();

         

        long startTime = System.currentTimeMillis();

        StatementHandler statementHandler = (StatementHandler)target;

        try {

            return invocation.proceed();

        } finally {

            long endTime = System.currentTimeMillis();

            long sqlCost = endTime - startTime;

             

            BoundSql boundSql = statementHandler.getBoundSql();

            String sql = boundSql.getSql();

            Object parameterObject = boundSql.getParameterObject();

            List parameterMappingList = boundSql.getParameterMappings();

             

            // 格式化Sql语句,去除换行符,替换参数

            sql = formatSql(sql, parameterObject, parameterMappingList);

             

            System.out.println('SQL:[' + sql + ']执行耗时[' + sqlCost + 'ms]');

        }

    }

 

    @Override

    public Object plugin(Object target) {

        return Plugin.wrap(target, this);

    }

 

    @Override

    public void setProperties(Properties properties) {

         

    }

     

    @SuppressWarnings('unchecked')

    private String formatSql(String sql, Object parameterObject, List parameterMappingList) {

        // 输入sql字符串空判断

        if (sql == null || sql.length() == 0) {

            return '';

        }

         

        // 美化sql

        sql = beautifySql(sql);

         

        // 不传参数的场景,直接把Sql美化一下返回出去

        if (parameterObject == null || parameterMappingList == null || parameterMappingList.size() == 0) {

            return sql;

        }

         

        // 定义一个没有替换过占位符的sql,用于出异常时返回

        String sqlWithoutReplacePlaceholder = sql;

         

        try {

            if (parameterMappingList != null) {

                Class parameterObjectClass = parameterObject.getClass();

 

                // 如果参数是StrictMap且Value类型为Collection,获取key='list'的属性,这里主要是为了处理循环时传入List这种参数的占位符替换

                // 例如select * from xxx where id in ...

                if (isStrictMap(parameterObjectClass)) {

                    StrictMap> strictMap = (StrictMap>)parameterObject;

                     

                    if (isList(strictMap.get('list').getClass())) {

                        sql = handleListParameter(sql, strictMap.get('list'));

                    }

                } else if (isMap(parameterObjectClass)) {

                    // 如果参数是Map则直接强转,通过map.get(key)方法获取真正的属性值

                    // 这里主要是为了处理

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多