首页 > 数据库 >mybatis 通过拦截器打印完整的sql语句以及执行结果操作

mybatis 通过拦截器打印完整的sql语句以及执行结果操作

时间:2024-02-08 10:45:00浏览次数:36  
标签:拦截器 String sql Object org mybatis import apache

  1. 下面的文件放在source-fw的【jp.co.token.sikyuu.iterceptor】包下面
  • MybatisInterceptor.java

  • InterceptorForQry.java

  • 下面的文件放在source-fw的【jp.co.token.sikyuu.common】包下面

  • FastJsonUtils.java

  • 下面的文件放在WebRoot/WEB-INF/lib/ 路径下面

  • fastjson-1.2.76.jar

  • 给source里面的【myBatis-config.xml】文件里面,添加下面的内容

具体添加的位置,参照下记的图片

img

MybatisInterceptor.java

package jp.co.token.SIKYUU.iterceptor;

import java.text.DateFormat;
import java.util.Date;
import java.util.List;
import java.util.Locale;
import java.util.Properties;
import java.util.regex.Matcher;
import org.apache.ibatis.executor.Executor;
import org.apache.ibatis.mapping.BoundSql;
import org.apache.ibatis.mapping.MappedStatement;
import org.apache.ibatis.mapping.ParameterMapping;
import org.apache.ibatis.plugin.Interceptor;
import org.apache.ibatis.plugin.Intercepts;
import org.apache.ibatis.plugin.Invocation;
import org.apache.ibatis.plugin.Plugin;
import org.apache.ibatis.plugin.Signature;
import org.apache.ibatis.reflection.MetaObject;
import org.apache.ibatis.session.Configuration;
import org.apache.ibatis.session.ResultHandler;
import org.apache.ibatis.session.RowBounds;
import org.apache.ibatis.type.TypeHandlerRegistry;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* <pre>
* [功 能] 完整的SQL语句输出
* </pre>
*/
@Intercepts({
       @Signature(type = Executor.class, method = "update", args = { MappedStatement.class,
               Object.class }),
       @Signature(type = Executor.class, method = "query", args = { MappedStatement.class,
               Object.class, RowBounds.class, ResultHandler.class }) })
@SuppressWarnings({})
public class MybatisInterceptor implements Interceptor {
   /**
    * 记录对象
    */
   protected Logger log = LoggerFactory.getLogger(MybatisInterceptor.class);
   @Override
   public Object intercept(Invocation invocation)
           throws Throwable {
       try {
           // 获取xml中的一个select/update/insert/delete节点,是一条SQL语句
           MappedStatement mappedStatement = (MappedStatement) invocation.getArgs()[0];
           Object parameter = null;
           // 获取参数,if语句成立,表示sql语句有参数,参数格式是map形式
           if (invocation.getArgs().length > 1) {
               parameter = invocation.getArgs()[1];
               log.debug("パラメータデータ: " + parameter);
           }
           // 获取到节点的id, 即sql语句的id
           String sqlId = mappedStatement.getId();
           log.debug("実行SQLID: " + sqlId);
           // BoundSql就是封装myBatis最终产生的sql类
           BoundSql boundSql = mappedStatement.getBoundSql(parameter);
           // 获取节点的配置
           Configuration configuration = mappedStatement.getConfiguration();
           // 获取到最终的sql语句
           String sql = getSql(configuration, boundSql, "");
           log.debug("sql = " + sql);
       } catch (Exception e) {
           e.printStackTrace();
       }
       return invocation.proceed();
   }
   // 封装了一下sql语句,使得结果返回完整xml路径下的sql语句节点id + sql语句
   public static String getSql(Configuration configuration, BoundSql boundSql, String sqlId) {
       String sql = showSql(configuration, boundSql);
       StringBuilder str = new StringBuilder(100);
       str.append(sqlId);
       str.append(":");
       str.append(sql);
       return str.toString();
   }
   // 如果参数是String,则添加单引号, 如果是日期,则转换为时间格式器并加单引号; 对参数是null和不是null的情况作了处理
   private static String getParameterValue(Object obj) {
       String value = null;
       if (obj instanceof String) {
           value = "'" + obj.toString() + "'";
       } else if (obj instanceof Date) {
           DateFormat formatter = DateFormat.getDateTimeInstance(DateFormat.DEFAULT,
                   DateFormat.DEFAULT, Locale.CHINA);
           value = "'" + formatter.format(new Date()) + "'";
       } else {
           if (obj != null) {
               value = obj.toString();
           } else {
               value = "";
           }
       }
       return value;
   }
   // 进行?的替换
   public static String showSql(Configuration configuration, BoundSql boundSql) {
       // 获取参数
       Object parameterObject = boundSql.getParameterObject();
       List<ParameterMapping> parameterMappings = boundSql.getParameterMappings();
       // sql语句中多个空格都用一个空格代替
       String sql = boundSql.getSql().replaceAll("[\\s]+", " ");
       if ((parameterMappings != null && !parameterMappings.isEmpty()) && parameterObject != null) {
           // 获取类型处理器注册器,类型处理器的功能是进行java类型和数据库类型的转换
           TypeHandlerRegistry typeHandlerRegistry = configuration.getTypeHandlerRegistry();
           // 如果根据parameterObject.getClass()可以找到对应的类型,则替换
           if (typeHandlerRegistry.hasTypeHandler(parameterObject.getClass())) {
               sql = sql.replaceFirst("\\?",
                       Matcher.quoteReplacement(getParameterValue(parameterObject)));
           } else {
               // MetaObject主要是封装了originalObject对象,提供了get和set的方法用于获取和设置originalObject的属性值,主要支持对JavaBean、Collection、Map三种类型对象的操作
               MetaObject metaObject = configuration.newMetaObject(parameterObject);
               for (ParameterMapping parameterMapping : parameterMappings) {
                   String propertyName = parameterMapping.getProperty();
                   if (metaObject.hasGetter(propertyName)) {
                       Object obj = metaObject.getValue(propertyName);
                       sql = sql.replaceFirst("\\?",
                               Matcher.quoteReplacement(getParameterValue(obj)));
                   } else if (boundSql.hasAdditionalParameter(propertyName)) {
                       // 该分支是动态sql
                       Object obj = boundSql.getAdditionalParameter(propertyName);
                       sql = sql.replaceFirst("\\?",
                               Matcher.quoteReplacement(getParameterValue(obj)));
                   } else {
                       // 打印出缺失,提醒该参数缺失并防止错位
                       sql = sql.replaceFirst("\\?", "欠ける");
                   }
               }
           }
       }
       return sql;
   }
   @Override
   public Object plugin(Object target) {
       return Plugin.wrap(target, this);
   }
   @Override
   public void setProperties(Properties properties) {
   }
}

InterceptorForQry.java

package jp.co.token.SIKYUU.iterceptor;

import java.util.Properties;

import jp.co.token.SIKYUU.common.FastJsonUtils;
import org.apache.ibatis.executor.Executor;
import org.apache.ibatis.mapping.MappedStatement;
import org.apache.ibatis.plugin.Interceptor;
import org.apache.ibatis.plugin.Intercepts;
import org.apache.ibatis.plugin.Invocation;
import org.apache.ibatis.plugin.Plugin;
import org.apache.ibatis.plugin.Signature;
import org.apache.ibatis.session.ResultHandler;
import org.apache.ibatis.session.RowBounds;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

@Intercepts({ @Signature(type = Executor.class, method = "query", args = { MappedStatement.class,
        Object.class, RowBounds.class, ResultHandler.class }) })
public class InterceptorForQry implements Interceptor {
    /**
     * 记录对象
     */
    protected Logger log = LoggerFactory.getLogger(InterceptorForQry.class);
    @SuppressWarnings({})
    public Object intercept(Invocation invocation)
            throws Throwable {
        Object result = invocation.proceed();
        String str = FastJsonUtils.toJSONString(result);
        log.debug("SQL执行结果: " + str);
        return result;
    }
    public Object plugin(Object target) {
        return Plugin.wrap(target, this);
    }
    public void setProperties(Properties arg0) {
    }
}

FastJsonUtils.java

package jp.co.token.SIKYUU.common;

import java.util.List;
import java.util.Map;
import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.JSONObject;
import com.alibaba.fastjson.serializer.JSONLibDataFormatSerializer;
import com.alibaba.fastjson.serializer.SerializeConfig;
import com.alibaba.fastjson.serializer.SerializerFeature;
/**
 * <pre>
 * [功 能] fastjson
 * </pre>
 */
public class FastJsonUtils {
    private static final SerializeConfig config;
    static {
        config = new SerializeConfig();
        config.put(java.util.Date.class, new JSONLibDataFormatSerializer());
        config.put(java.sql.Date.class, new JSONLibDataFormatSerializer());
    }
    private static final SerializerFeature[] features = {SerializerFeature.WriteMapNullValue,
            SerializerFeature.WriteNullListAsEmpty,
            SerializerFeature.WriteNullNumberAsZero,
            SerializerFeature.WriteNullBooleanAsFalse,
            SerializerFeature.WriteNullStringAsEmpty
    };
    public static String toJSONString(Object object) {
        return JSON.toJSONString(object, config, features);
    }
    public static String toJSONNoFeatures(Object object) {
        return JSON.toJSONString(object, config);
    }
    public static Object toBean(String text) {
        return JSON.parse(text);
    }
    public static <T> T toBean(String text, Class<T> clazz) {
        return JSON.parseObject(text, clazz);
    }
    public static <T> Object[] toArray(String text) {
        return toArray(text, null);
    }
    public static <T> Object[] toArray(String text, Class<T> clazz) {
        return JSON.parseArray(text, clazz).toArray();
    }
    public static <T> List<T> toList(String text, Class<T> clazz) {
        return (List<T>) JSON.parseArray(text, clazz);
    }
    /*public static Object beanToJson(KeyValue keyvalue) {
        String textJson = JSON.toJSONString(keyvalue);
        Object objectJson  = JSON.parse(textJson);
        return objectJson;
    }*/
    public static Object textToJson(String text) {
        Object objectJson  = JSON.parse(text);
        return objectJson;
    }
    @SuppressWarnings("rawtypes")
    public static Map stringToCollect(String s) {
        Map m = JSONObject.parseObject(s);
        return m;
    }
    @SuppressWarnings("rawtypes")
    public static String collectToString(Map m) {
        String s = JSONObject.toJSONString(m);
        return s;
    }
}

[fastjson-1.2.76.jar]:

<!-- https://mvnrepository.com/artifact/com.alibaba/fastjson -->
<dependency>
    <groupId>com.alibaba</groupId>
    <artifactId>fastjson</artifactId>
    <version>1.2.76</version>
</dependency>

标签:拦截器,String,sql,Object,org,mybatis,import,apache
From: https://www.cnblogs.com/ice204/p/18011638

相关文章

  • MySQL优化
    优化分为六大部分:SQL语句的优化索引的优化表结构的优化事务优化系统配置优化物理机的优化SQL语句的优化a.尽量使用select字段名,不要使用select*,select*不能使用索引覆盖。只查需要用到的列。b.小表驱动大表。主查询in/exists子查询.ⅰ.in先执行右边的子查询......
  • 【转帖】数据库传奇:MySQL创世之父的两千金My、Maria
    https://zhuanlan.zhihu.com/p/672142719 1人赞同了该文章《数据库传奇:MySQL创世之父的两千金My、Maria》一、前言  MySQL是一款备受欢迎的关系型数据库管理系统(RDBMS),最初由瑞典公司MySQLAB开发,目前隶属于OracleCorporation。在DB-Engines的排名中,MySQL稳......
  • 后台没有调用分页方法,但是sql的执行却走了分页
    1、分页方法主要调用PageHelper.startPage()现象:开始是可以查询所有数据,在我调整前台代码以后,SQl查询就变为分页的了。因此,肯定是vue代码有问题,具体为啥前台代码会影响后台的分页呢?很是神奇。(计算机问题,表面上都有神秘的面纱哈哈哈)做法:(1)比对请求参数,发现比原来多了分页参数pa......
  • Java与sql中的字符串表示
    在Java中,双引号""用于表示字符串字面量,而单引号''用于表示字符字面量。这意味着在Java中,您可以使用双引号来包围包含任意数量字符的字符串,包括零个字符(空字符串)和多个字符。例如,在Java中:StringemptyString="";//空字符串StringsingleChar='a';/......
  • MyBatis缓存
    MyBatis缓存缓存就是内存中的数据,常常来自对数据库查询结果的保存,使用缓存,我们可以避免频繁的与数据进行交互,进而提高响应速度。MyBatis也提供两种缓存模式,分为一级缓存和二级缓存。优点:提供查询效率减少频繁进行I/O操作,从而减少数据库的压力。适合存放缓存的数据:【1】查询频......
  • docker-compose部署nacos2.3.0+mysql5.7
    docker-compose-nacos.ymlservices:mysql-nacos:restart:alwaysimage:mysql:5.7.18container_name:mysql-lablevolumes:-/Users/docker/mysql/source/mydir:/mydir-/Users/docker/mysql/source/datadir:/var/lib/mysql-/U......
  • Mybatis Plus java.lang.NoSuchMethodError: com.baomidou.mybatisplus.core.toolkit.
    问题描述在进行SpringBoot整合MybatisPlus时提示10:49:08.390[restartedMain]DEBUGorg.springframework.boot.context.logging.ClasspathLoggingApplicationListener-Applicationfailedtostartwithclasspath:[file:/D:/%e7%99%be%e5%ba%a6%e7%bd%91%e7%9b%98/Vue......
  • centos7安装posgresql
    安装#安装源sudoyuminstall-yhttps://download.postgresql.org/pub/repos/yum/reporpms/EL-7-x86_64/pgdg-redhat-repo-latest.noarch.rpm#安装PostgreSQLcentos7上安装更高版本缺依赖,比较麻烦sudoyuminstall-ypostgresql14-server#初始化数据库sudo/usr/......
  • MySQL 根据条件迁移数据
    在实际数据库管理和数据处理中,经常需要将数据从一个表迁移到另一个表,或者在同一个表中根据某些条件进行数据迁移。MySQL作为一个流行的关系型数据库管理系统,提供了多种方法来实现数据的迁移。本文将介绍如何使用MySQL来根据条件迁移数据的技术方法。1.使用INSERTINTOSELECT语句I......
  • ThinkPHP 6.0 SQL注入漏洞修复
    公司买的官网被政府网安检测出SQL注入漏洞:隐患描述SQL漏洞证明语句: python3sqlmap.py-u"http://xxxx?keywords=1"-pkeywords--level=5--risk=3--tamper=space2comment.py,between.py--current-db--random-agent--time-sec=10--batch--dbms="MySQL"漏洞回显:权......