MyBatis的SQL执行结果和客户端执行结果不一致问题排查

作者:modelmd 时间:2022-12-30 20:37:22 

最近遇到一个调试很久的问题,MyBatis 查询 Oracle 数据库查询结果与在客户端查询结果不一致。

问题引入

测试表、测试数据

创建测试表、序列

CREATE TABLE t_test_table (
"ID" NUMBER(18,0),
"CREATE_TIME" TIMESTAMP(6),
"FIELD_TYPE" CHAR(20),
CONSTRAINT pk_id PRIMARY KEY(ID)
)

CREATE SEQUENCE seq_t_test_table;

插入测试数据

INSERT INTO t_test_table ("ID", "CREATE_TIME", "FIELD_TYPE") VALUES (SEQ_T_TEST_TABLE.nextval, SYSDATE - NUMTODSINTERVAL(1, 'DAY'), 'Integer');
INSERT INTO t_test_table ("ID", "CREATE_TIME", "FIELD_TYPE") VALUES (SEQ_T_TEST_TABLE.nextval, SYSDATE - NUMTODSINTERVAL(1, 'HOUR'), 'Double');
INSERT INTO t_test_table ("ID", "CREATE_TIME", "FIELD_TYPE") VALUES (SEQ_T_TEST_TABLE.nextval, SYSDATE - NUMTODSINTERVAL(1, 'MINUTE'), 'Long');

查询数据是否插入成功

MyBatis的SQL执行结果和客户端执行结果不一致问题排查

问题介绍

MyBatis xml 配置

<select id="selectByFieldType" resultType="com.scd.model.po.TestTable">
       SELECT * FROM T_TEST_TABLE WHERE FIELD_TYPE = #{filedType}
</select>

运行输出的日志

20:26:08.678 [main] DEBUG com.scd.mapper.OracleBatchSeqMapper.selectByFieldType - ==>  Preparing: SELECT * FROM T_TEST_TABLE WHERE FIELD_TYPE = ?
20:26:08.906 [main] DEBUG com.scd.mapper.OracleBatchSeqMapper.selectByFieldType - ==> Parameters: Double(String)
20:26:09.013 [main] DEBUG com.scd.mapper.OracleBatchSeqMapper.selectByFieldType - <==      Total: 0

这里显示输出的结果为0条,之前有插入数据的,明显出错了,于是把sql语句复制到DBeaver客户端执行的时候,是可以查询数据的

MyBatis的SQL执行结果和客户端执行结果不一致问题排查

排查问题

日志中的SQL 和 客户端的 SQL 一致的,唯一不同的就是日志中是使用占位符形式,也就是预编译的SQL, 而客户端直接执行的SQL。首先第一步就是把xml中的预编译SQL修改成字符串拼接的形式

<select id="selectByFieldType" resultType="com.scd.model.po.TestTable">
       SELECT * FROM T_TEST_TABLE WHERE FIELD_TYPE = '${filedType}'
   </select>

执行输出的结果确实有一条,和客户端的一致,运行日志结果如下

20:38:45.603 [main] DEBUG com.scd.mapper.OracleBatchSeqMapper.selectByFieldType - ==>  Preparing: SELECT * FROM T_TEST_TABLE WHERE FIELD_TYPE = 'Double'
20:38:45.747 [main] DEBUG com.scd.mapper.OracleBatchSeqMapper.selectByFieldType - ==> Parameters: 
20:38:45.844 [main] DEBUG com.scd.mapper.OracleBatchSeqMapper.selectByFieldType - <==      Total: 1

用这种方式虽然可以解决,但是很多编码规范在 xml 中不允许配置 $,防止SQL注入。还需要找一下为啥 # 的形式不能得到正确的结果。

调试 MyBatis源码

对比拼接 SQL 和 预编译的 SQL,区别在于设置参数填充,找到 MyBatis中这一块的执行逻辑,经过调试,定位到设置参数的代码在 org.apache.ibatis.scripting.defaults.DefaultParameterHandler#setParameters

MyBatis的SQL执行结果和客户端执行结果不一致问题排查

由于xml中未指定TypeHandler,程序使用默认的 UnknownTypeHandler。根据参数的类型去匹配TypeHandler

MyBatis的SQL执行结果和客户端执行结果不一致问题排查

参数 &ldquo;Double&rdquo; 是字符串,匹配到了 StringTypeHandler

MyBatis的SQL执行结果和客户端执行结果不一致问题排查

继续调试,发现设置参数的代码段如下

MyBatis的SQL执行结果和客户端执行结果不一致问题排查

发现整个设置数据的过程没有啥问题呀,于是把问题简化一下,弄成JDBC的方式执行看看。

JDBC 执行 SQL

按照 MyBtatis 的执行过程,把代码简化成如下

import org.junit.Test;
import java.sql.*;
import java.util.Properties;

/**
* @author James
* @date 2022/12/10 19:02
*/
public class OracleJdbc {

private static final String driver = "oracle.jdbc.driver.OracleDriver";

private static final String url = "jdbc:oracle:thin:@localhost:1521/TEST";

private static final String userName = "TEST_USER";

private static final String password = "TEST_USER";

static {
       // 加载驱动
       try {
           Class.forName(driver);
       } catch (ClassNotFoundException e) {
           e.printStackTrace();
       }
   }

public Connection createConnection(String url, String userName, String password) {
       Connection connection = null;
       try {
           connection = DriverManager.getConnection(url, userName, password);
       } catch (SQLException e) {
           e.printStackTrace();
       }
       return connection;
   }

@Test
   public void testStrQuery() {
       try (Connection connection = createConnection(url, userName, password)) {
           String sql = "SELECT * FROM T_TEST_TABLE WHERE FIELD_TYPE = ?";
           PreparedStatement preparedStatement = connection.prepareStatement(sql);
           preparedStatement.setString(1, "Double");
           preparedStatement.execute();
           while (preparedStatement.getMoreResults()) {
               System.out.println(preparedStatement.getResultSet());
           }
       } catch (SQLException e) {
           e.printStackTrace();
       }
   }
}

运行测试用例,发现使用JDBC也无法获取到正确的结果。于是打算看看Oracle的SQL执行日志,看预编译的SQL与直接拼接的SQL有啥区别

select * from  v$sql WHERE SQL_TEXT LIKE '%T_TEST_TABLE%'
ORDER BY FIRST_LOAD_TIME DESC;

MyBatis的SQL执行结果和客户端执行结果不一致问题排查

根据运行的时间点,找到预编译的 SQL为

SELECT * FROM T_TEST_TABLE WHERE FIELD_TYPE = :1

直接拼接的SQL为

SELECT * FROM T_TEST_TABLE WHERE FIELD_TYPE = 'Double'

预编译的SQL在客户端中运行也没问题

MyBatis的SQL执行结果和客户端执行结果不一致问题排查

说明代码没问题,是不是字段里面有空格导致查询不到的呢?于是把字段直接复制出来。复制出来果然发现了空格

MyBatis的SQL执行结果和客户端执行结果不一致问题排查

但是运行这个也不对。这个时候思考着为啥有空格呢?于是百度看了一下CHAR字段的介绍,原来Oracle的 CHAR字段长度不够的用空格填充,输出上面的字符长度看看

MyBatis的SQL执行结果和客户端执行结果不一致问题排查

解决问题

由于Oracle的CHAR类型在插入的数据长度不够的情况下会用空格填充,于是把字段类型修改成 VARCHAR2

ALTER TABLE T_TEST_TABLE MODIFY "FIELD_TYPE" VARCHAR2(20)

把之前的数据空格清除

UPDATE T_TEST_TABLE SET FIELD_TYPE = TRIM(FIELD_TYPE)

再次执行 MyBatis 的测试方法,可以发现查出数据

MyBatis的SQL执行结果和客户端执行结果不一致问题排查

待解决问题

为啥用JDBC 预编译 SQL 查询 CHAR 类型的字段会有空格匹配问题?

来源:https://blog.csdn.net/modelmd/article/details/128460956

标签:MyBatis,SQL执行结果,客户端执行结果
0
投稿

猜你喜欢

  • Java常用集合与原理解析

    2023-04-01 14:26:42
  • Java 中很好用的数据结构EnumSet

    2023-12-06 09:37:23
  • SpringBoot实现本地文件存储及预览过程

    2022-07-31 05:30:58
  • 基于ReentrantLock的实现原理讲解

    2023-11-23 22:43:23
  • 浅谈android nexus私服的使用

    2022-09-24 01:30:59
  • 高并发下restTemplate的错误分析方式

    2023-08-23 12:58:24
  • Android Fragment+FragmentTabHost组件实现常见主页面(仿微信新浪)

    2023-10-20 10:00:50
  • Java使用Tess4J实现图像识别方式

    2022-10-07 19:24:11
  • Android自定义控件之组合控件学习笔记分享

    2022-09-18 01:09:22
  • Android开发升级AGP7.0后的一些适配方法技巧

    2022-10-22 17:41:22
  • Android的HTTP类库Volley入门学习教程

    2022-06-08 11:56:51
  • C# SqlHelper应用开发学习

    2023-05-07 11:07:35
  • 介绍C# 泛型类在使用中约束

    2023-06-23 21:39:40
  • Java设计模式之模板方法模式详解

    2021-08-04 04:32:51
  • Java创建型设计模式之抽象工厂模式(Abstract Factory)

    2022-08-31 01:25:56
  • C#基于Socket实现简单聊天室功能

    2023-05-09 01:17:32
  • IntelliJ IDEA 如何配置git的操作方法

    2021-12-28 11:24:44
  • Java获取用户IP属地模拟抖音详解

    2023-04-18 02:01:29
  • 分布式系统下调用链追踪技术面试题

    2023-11-25 05:55:00
  • C#线程倒计时器源码分享

    2023-08-16 07:23:36
  • asp之家 软件编程 m.aspxhome.com