背景:

在我日常码代码的时候,由于对mybatis的动态sql,比较依赖,并且有时候需求复杂,导致sql较长,而且参数众多,当出现问题是,需要将sql,放到navicat里面去执行查看结果,但是对于复杂的sql来说,众多的参数,一个一个替换。当真很麻烦,于是萌生出可不可以将sql直接输出,不在出线sql和参数分开的情况,可以减少很多麻烦,在我找度娘,一次又一次的尝试,我还是没有发现,在log4j的配置文件里面。有这个功能,所以最后萌生出改写源码的想法,之后我也会尝试继续寻找,有没有官方的API提供,本文讲述我自己改写源代码实现的方案
一、针对DEMO搭建SSM项目(略)
二、采用的log4j的版本(这里由于log的众多实现,有可能出线log4j的冲突,而且我也没有滤的特别清楚,所以贴出自己使用的log4j的版本,采用maven提供)
  1. <span style="white-space:pre;">         </span><!-- log4j 日志 -->
  2. <dependency>
  3. <groupId>log4j</groupId>
  4. <artifactId>log4j</artifactId>
  5. <version>${log4j-version}</version>
  6. </dependency>
三、首先实现输出mybatis输出sql
1、引入jar包(同上)
2、配置mybatis输出sql的配置项
  1. <span style="white-space:pre;"> </span><settings>
  2. <setting name="logImpl" value="LOG4J" />
  3. </settings>

3、添加logj4j.properties文件

  1. #logFile
  2. log4j.rootLogger=DEBUG,Console
  3. #Console
  4. log4j.appender.Console=org.apache.log4j.ConsoleAppender
  5. log4j.appender.console.Threshold=INFO
  6. log4j.appender.console.ImmediateFlush=true
  7. log4j.appender.Console.Target=System.out
  8. log4j.appender.Console.layout=org.apache.log4j.PatternLayout
  9. log4j.appender.Console.layout.ConversionPattern=%-5p %d{yyyy-MM-dd HH:mm:ss}[%t] %l: %x%m%n
  10. # 日志文件(logFile)
  11. log4j.appender.logFile=org.apache.log4j.FileAppender
  12. log4j.appender.logFile.Threshold=DEBUG
  13. # 立即输出
  14. #log4j.appender.logFile.ImmediateFlush=true
  15. #log4j.appender.logFile.Append=true
  16. #log4j.appender.logFile.File=D:/logs/log.log4j
  17. #log4j.appender.logFile.layout=org.apache.log4j.PatternLayout
  18. #log4j.appender.logFile.layout.ConversionPattern=%-5p %d{yyyy-MM-dd HH:mm:ss}[%t] %l: %x%m%n
  19. log4j.logger.org.apache=ERROR
  20. log4j.logger.org.mybatis=ERROR
  21. log4j.logger.org.springframework=ERROR
  22. #这个需要
  23. log4j.logger.log4jdbc.debug=ERROR
  24. log4j.logger.jdbc.audit=ERROR
  25. log4j.logger.jdbc.resultset=ERROR
  26. #这个打印SQL语句非常重要
  27. log4j.logger.jdbc.sqlonly=DEBUG
  28. log4j.logger.jdbc.sqltiming=ERROR
  29. log4j.logger.jdbc.connection=FATAL

4、测试结果

  1. ==>  Preparing: select u.id as id ,u.name as name , u.pwd as pwd from user u where u.pwd = ? and u.id = ?;
  2. ==> Parameters: 123(String), 1(String)
  3. <==      Total: 0

显然:参数和sql是分离的,现在要实现将参数嵌套进sql中,实现上面的想法

 
通过源码的分析最后我决定动(org.apache.ibatis.logging.jdbc.BaseJdbcLogger)这个类,它的作用是输出最后的sql
 
方法步骤:
 
1、找到这个类的全路径,找到源码
2、在src下创建一个同路径下的BaseJdbcLogger类,并且将源码原封不动的贴近刚创建的类,
3、找到需要修改的方法,我贴一下我准备修改的方法
  1. protected void debug(String text, boolean input) {
  2. if (statementLog.isDebugEnabled()) {
  3. statementLog.debug(prefix(input) + text);
  4. }
  5. }

目前的这个是BaseJdbcLogger.debug的源码,其中(text:是需要打印的文本,input:表示前面的“==>”的方向)

经过我的修改:
  1. /* 打印的sql */
  2. private static String sql = "";
  3. protected void debug(String text, boolean input) {
  4. text = text.trim();
  5. if (statementLog.isDebugEnabled()) {
  6. if(text.startsWith("Preparing:")){
  7. sql = text.substring(text.indexOf(":")+1);
  8. return ;
  9. }
  10. if(text.startsWith("Parameters:")){
  11. String temp = text.substring(text.indexOf(":")+1);
  12. String[] split = temp.split(",");
  13. if(split != null & split.length > 0){
  14. for (String string2 : split) {
  15. String s = string2.trim();
  16. sql = sql.replaceFirst("\\?", s.substring(0, s.indexOf("(")));
  17. }
  18. }
  19. text = "Preparing:"+ sql ;
  20. sql = "";
  21. }
  22. statementLog.debug(prefix(input) + text);
  23. }
  24. }

最终输出结果:

  1. ==> Preparing: select u.id as id ,u.name as name , u.pwd as pwd from user u where u.pwd = 123 and u.id = 1;
  2. <== Total: 0

到到最终目的

(虽然不是最好的方法,mybatis应该分装相应的配置,但是我没有发现,如果有码友发现了,请告知我,谢谢)

log4j打印mybatis执行sql,将占位符换成真实的参数输出的更多相关文章

  1. mybatis + log4j 打印mybatis的sql

    项目中使用log4j管理日志,同时使用了mybatis 在log4j中rootLogger级别是info的情况下正常是不会打印sql出来的,这个时候设置如下: log4j.rootLogger=inf ...

  2. log4j打印MyBatis的sql语句配置

    log4j.rootLogger=DEBUG,stdout log4j.appender.stdout=org.apache.log4j.ConsoleAppender log4j.appender. ...

  3. log4j打印mybatis sql语句

    Mybatis默认使用有slf4j 必须加上依赖 <dependency> <groupId>org.slf4j</groupId> <artifactId& ...

  4. linux 的sed命令解释 sed ':t;N;s/\n/,/;b t' 将换行符换成逗号

    linux 的sed命令解释 sed ':t;N;s/\n/,/;b t' 将换行符换成逗号 实现的功能是吧换行符换成逗号了,自己试验过. 求解释,:t N b t 都是什么意思??? :t 定义la ...

  5. log4j配置打印mybatis的sql到控制台(复制)

    log4j.rootLogger=DEBUG, stdout log4j.appender.stdout=org.apache.log4j.ConsoleAppender log4j.appender ...

  6. Mybatis执行SQL的流程

    前篇:Mybatis初始化过程 SqlSession : SqlSession是一个接口,它有两个实现类:DefaultSqlSession (默认)和 SqlSessionManager (弃用,不 ...

  7. Mybatis执行sql(insert、update、delete)返回值问题

    数据库:Mysql 在使用mybatis的过程中对执行sql的返回值产生疑问,顺手记录一下. 结论: insert:   插入n条记录,返回影响行数n.(n>=1,n为0时实际为插入失败) up ...

  8. 在mybatis执行SQL语句之前进行拦击处理

    转载自:http://blog.csdn.net/hfmbook/article/details/41985853 比较适用于在分页时候进行拦截.对分页的SQL语句通过封装处理,处理成不同的分页sql ...

  9. 如何让SpringBoot工程在log/控制台中实时打印MyBatis执行的SQL语句

    工程下载:https://files.cnblogs.com/files/xiandedanteng/gatling20200429-4.zip 其实就是一句话设置的事情,实现步骤: 在applica ...

随机推荐

  1. 15.并发容器之ConcurrentLinkedQueue

    1.ConcurrentLinkedQueue简介 在单线程编程中我们会经常用到一些集合类,比如ArrayList,HashMap等,但是这些类都不是线程安全的类.在面试中也经常会有一些考点,比如Ar ...

  2. @Primary-在spring中常被忽视的注解

    在spring 中使用注解,常使用@Autowired, 默认是根据类型Type来自动注入的.但有些特殊情况,对同一个接口,可能会有几种不同的实现类,而默认只会采取其中一种的情况下 @Primary  ...

  3. Vlmcsd(KMS)激活服务器程序

    1.下载vlmcsd程序 2-1.虚拟机版本: 新建Linux虚拟机,硬件仅保留内存(最小14MB,推荐16MB).处理器(1个1核心).软盘(指向floppy144.flp).网络适配器(桥接模式) ...

  4. day32 Python与金融量化分析(二)

    第一部分:金融与量化投资 股票: 股票是股份公司发给出资人的一种凭证,股票的持有者就是股份公司的股东. 股票的面值与市值 面值表示票面金额 市值表示市场价值 上市/IPO: 企业通过证券交易所公开向社 ...

  5. Ansible 小手册系列 三(命令介绍)

    仅仅只是介绍,可以选择跳过 ansible ansible是指令核心部分,其主要用于执行ad-hoc命令,即单条命令.默认后面需要跟主机和选项部分,默认不指定模块时,使用的是command模块. Us ...

  6. 023——VUE中数据排序sort() / 数据反转 reverse() 的使用

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  7. C++高级编程2. 静态动态链接库

    C++高级编程2. 静态动态链接库20131018 1.动态链接库和静态链接库的区别: 静态链接库就是把lib文件中用到的函数代码直接连接进目标程序,程序运行的时候不在需要其他的库文件:动态链接库是把 ...

  8. git clone 提示输入git@xxx的密码

    如下: suse:~/ecox # git clone git@vcs.in.ww-it.cn:ecox/ecox.git 正克隆到 'ecox'... git@vcs.in.ww-it.cn's p ...

  9. Linux:wc命令详解

    wc 用来计算数字 利用wc指令我们可以计算文件的Byte数.字数或是列数,若不指定文件名称,或是所给予的文件名为“-”,则wc指令会从标准输入设备读取数据. 语法 wc(选项)(参数) 选项 -c或 ...

  10. 使用jQuery操作DOM(1)

    1.常见方法 css(“属性”,”属性值”); //设置单个样式 css({属性1:属性值1,属性2:属性值3...}); //设置多个样式 addClass(“样式名”); //追加单个样式 add ...