mybatis中<if>条件判断带数字的字符串失效问题

简介: MySQL数据库使用Mybatis查询拼接select语句中进行`<if>条件拼接`的时候,发现带数字的或者带单个字母的字符串失效问题。举例说明:我`Log对象`有个属性`accountId是字符串类型`,假设我给它赋值为“1”,按常理sql拼接的`应该是and account_name = 'unmadmin'` ,然而实际判断拼接的`却是and account_name != 'unmadmin'`,明显感觉问题出在这里`and log.accountId == '1'` 这里校验失败了,或者说失效了,估计很多人会踩坑,那么为什么呢?不应该啊?所以此刻就是为了定位它原因。

1.jpeg

一、项目背景

MySQL数据库使用Mybatis查询拼接select语句中进行<if>条件拼接的时候,发现带数字的或者带单个字母的字符串失效问题。
举例说明:我Log对象有个属性accountId是字符串类型,假设我给它赋值为“1”,按常理sql拼接的应该是and account_name = 'unmadmin' ,然而实际判断拼接的却是and account_name != 'unmadmin',明显感觉问题出在这里and log.accountId == '1' 这里校验失败了,或者说失效了,估计很多人会踩坑,那么为什么呢?不应该啊?所以此刻就是为了定位它原因。

Log 对象

public class Log {
   
    private String logId;
    private String accountId;
    private String accountName;
}

xml

<select id="queryOperateLog" resultType="Log">
    select * from user 
    <where>
            1 = 1
            <if test="log.accountId != null and log.accountId != '' and log.accountId == '1'">
                and account_name = 'unmadmin'
            </if>
            <if test="log.accountId != null and log.accountId != '' and log.accountId != '1'">
                and account_name != 'unmadmin'
            </if>
   </where>
   order by exectime desc
 </select>

二、真实错误原因说明

结论:mybatis会把纯数字的字符串进行强制转换,就是大白话讲说碰到log.accountId == '1'它会把'1'强制抓换成字符格式,而字符串“1” = = 字符'1',比较结果为false。
重点:Mybatis是使用的OGNL表达式来进行解析的,在OGNL的表达式中,'1'会被解析成字符,因为java是强类型的,char 和 一个String 会导致不等。所以if标签中的sql不会被解析。

那么这时候网友会问了,单引号''""双引号在mysql中感觉差不多啊,平常使用也没问题啊,其实主要区别在这里,国内chatGPT给出的答案是:
image.png

国外chatGPT给出的答案是:
image.png

那么网友会问了,平常我使用的时候输入字符串使用单引号''或者双引号""也没碰到过这个问题啊,其实问题出在String accountId值上:

  • 如果accountId值是多个字符串字母或者字符串和数字的组合,单引号双引号完全没问题。
  • 如果accountId值是纯数字,比如'1' '211'之类的,就会出现问题。

三、解决方案

既然原因已经找到了,那么如何解决它呢,其实有2种方案:

3.1针对纯数字的字符串值场景

方案1:外面使用单引号,里面使用双引号

错误写法:
image.png

正确写法:
image.png

方案2:使用.toString(),这个方案新颖,估计一般人见都没见过,即单引号保留,但是后面多拼接了个.toString()

image.png

3.2针对单个字符的字符串值场景

别以为只有纯数字有问题,实际单个字符也是有问题的,假设我有个String type字段,我给它判断值传字符y,具体如下

错误写法:你这么写判断也失效

<if test="type=='y'">
    and status = 0
</if>

正确写法:

<if test='type=="y"'>
    and status = 0
</if>

四、参考文献

备注说明:这东西不是我研究出来的,是恰巧搜到了两位道友的博客才有所了解的:↓
道友1博客:mybatis中条件判断带数字的字符串
道友2博客:Mybatis if 判断等于一个字符串

另外我实际追踪过源码,因为我猜测是获取对象值进行类型转换拼接sql的时候出现了判断问题,但是追着追着就跟踪丢了,所以源码究竟哪里进行转换的我也没找到,如果有道友找到了也可以分享出来一起学习下;

另外我甚至翻阅了Mybatis的OGNL表达式,但是也没找出个所以然来,我只知道accountId == '1'确实进行了类型转换,但是究竟是往字符转还是往数字类型转我也不知道,因为找不到官网做论证,所以这块如果有道友真正找到了也可以分享出来一起学习下。

完结、撒花!

目录
相关文章
mybatis-plus 判断参数是否为空并作为查询条件
mybatis-plus 判断参数是否为空并作为查询条件
1621 0
|
SQL Java 数据库连接
mybatis @Select注解中当参数为空则不添加该参数的判断
public interface OrderMapper extends SqlMapper{ @Select("select * from tbl_order where room like #{room} and mydate like #{my...
4643 0
|
Java 数据库连接 mybatis
mybatis异常 :元素内容必须由格式正确的字符数据或标记组成。
今天同事写一个查询接口的时候,出错:元素内容必须由格式正确的字符数据或标记组成。   错误原因:mybatis查询的时候,需要用到运算符 小于号:< 和  大于号: >,在mybatis配置文件里面,这种会被认为是标签,所以解析错误 错误事例: select from t...
1846 0
|
Java 数据库连接 mybatis
mybatis注解@Select中添加判断条件<script>
mybatis注解@Select中添加判断条件<script>
551 0
|
XML Java 数据库连接
mybatis返回整数值
mybatis返回整数值
83 0
|
SQL Java 数据库
【Mybatis-plus异常】mybatis-plus updateById方法更新不了空字符串和null的解决方法
【Mybatis-plus异常】mybatis-plus updateById方法更新不了空字符串和null的解决方法
675 0
|
7月前
|
Java 数据库连接 mybatis
mybatis plus字段为null或空字符串把原来的数据也更新了,只需要注解
mybatis plus字段为null或空字符串把原来的数据也更新了,只需要注解
225 0
|
7月前
|
Java 关系型数据库 MySQL
Mybatis和Mybatis-Plus执行插入语句后可以返回主键ID吗?
Mybatis和Mybatis-Plus执行插入语句后可以返回主键ID吗?
481 0
|
7月前
|
SQL Java 数据库连接
MyBatis【问题 02】使用#{}导致的问题(#{}并非简单地将传入数据前后添加单引号)及解决方法
MyBatis【问题 02】使用#{}导致的问题(#{}并非简单地将传入数据前后添加单引号)及解决方法
184 0
|
SQL Java 数据库连接
解决MyBatis判断整型失败的问题
解决MyBatis判断整型失败的问题
27 0