基于mybatis中test条件中单引号双引号的问题

 更新时间:2022年01月14日 09:32:11   作者:小生小生小小生  
这篇文章主要介绍了基于mybatis中test条件中单引号双引号的问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

test条件中单引号双引号问题

在mybatis中test判断条件中使用单引号会报错 通常使用双引号

通常test后的判断条件写在双引号内,但是当条件中判断使用字符串时应该如下方式开发

<when  test=“channel ==null” >
<when  test='channel =="QT"' >

具体原因

为单引号会被mybatis默认为字符类型,若为单字符可以使用单引号。否则会报错。 

动态sql中test的一些问题

mybatis动态sql中OGNL中type=="1"和type='1'的区别

最近在mybatis中使用OGNL所遇到的坑: 一点鸡肋: type=="1"和type='1'的区别

//CountryDao.java
public interface CountryDao {
    CountryDo getByType(@Param("type") String type);
}
//CountryManager.java
@Component("countryManager")
public class CountryManager {
    @Autowired
    private CountryDao countryDao;
    public CountryDo getByType(String type){
        return countryDao.getByType(type);
    }
}

mapper.xml:

//.xml
<select id="getByType" resultType="com.github.**.domain.CountryDo">
    <choose>
      //注意写法`type=='0'`
        <when test="type == '0'">
            SELECT * FROM country WHERE country_id = 2
        </when>
        <otherwise>
            SELECT * FROM country WHERE country_id = 3;
        </otherwise>
    </choose>
</select>

测试案例:

@Test
public void getByTypeTest() throws JsonProcessingException {
    String type = "0";
    CountryDo countryDo = countryManager.getByType(type);
    ObjectMapper mapper = new ObjectMapper();
    String json = mapper.writeValueAsString(countryDo);
    System.out.println(json);
}

理想情况下,当type=0时返回的是country_id=2的结果

测试结果:

{"countryId":3,"country":"American Samoa","lastUpdate":"2006-02-15 04:44:00.0"}

居然返回的是country_id=3的结果

百度了才知道

原来传值进去的是String类型,而mybatis的动态sql中的OGNL表达式不会将单引号包裹的char类型的内容转成String类型,并且String类型和char类型直接比较一定是不相等的。所以输出是country_id=3的结果

解决方案

将<when test="type == '0'">换成<when test='type == "0"'>即可,即将单引号换成双引号

0==’'问题

一个更有趣的问题,如果将mapper.xml文件内容换成:

<select id="getByType" resultType="com.github.sijing.domain.CountryDo">
    <choose>
        <when test="0 ==''">
            SELECT * FROM country WHERE country_id = 2
        </when>
        <otherwise>
            SELECT * FROM country WHERE country_id = 3;
        </otherwise>
    </choose>
</select>

测试案例不变:

@Test
public void getByTypeTest() throws JsonProcessingException {
    //此时传值已经没有影响了
    String type = "0";
    CountryDo countryDo = countryManager.getByType(type);
    ObjectMapper mapper = new ObjectMapper();
    String json = mapper.writeValueAsString(countryDo);
    System.out.println(json);
}

测试结果为:

{"countryId":2,"country":"Algeria","lastUpdate":"2006-02-15 04:44:00.0"}

怎么还是country_id=2的结果?,难道0==''??

查看mybatis源码发现,比如对于动态sql<if>标签的解析:

// IfSqlNode.
@Override
 public boolean apply(DynamicContext context) {
   if (evaluator.evaluateBoolean(test, context.getBindings())) {
     contents.apply(context);
     return true;
   }
   return false;
 }

在evaluator.evaluateBoolean方法中:

//ExpressionEvaluator.java OGNL表达式处理
public boolean evaluateBoolean(String expression, Object parameterObject) {
  Object value = OgnlCache.getValue(expression, parameterObject);
  if (value instanceof Boolean) {
    return (Boolean) value;
  }
  if (value instanceof Number) {
    return new BigDecimal(String.valueOf(value)).compareTo(BigDecimal.ZERO) != 0;
  }
  return value != null;
}

当传入值为0时,因为0是Number类型,所以被转成了BigDecimal类型,并与0作比较,test的结果为true,所以返回的是country_id=2的结果,这个说法有误。。。

再次调试,首先char ''被转成了字符String"",大概在OgnlOps.java文件的isEqual方法中的(compareWithConversion(object1, object2) == 0),点进去发现:

case NONNUMERIC:
    if ((t1 == NONNUMERIC) && (t2 == NONNUMERIC)) {
        if ((v1 instanceof Comparable) && v1.getClass().isAssignableFrom(v2.getClass())) {
            result = ((Comparable) v1).compareTo(v2);
            break;
        } else {
            throw new IllegalArgumentException("invalid comparison: " + v1.getClass().getName() + " and "
                    + v2.getClass().getName());
        }
    }
    // else fall through
case FLOAT:
case DOUBLE:
    //v1=0, v2=""
    double dv1 = doubleValue(v1),
    dv2 = doubleValue(v2);
    return (dv1 == dv2) ? 0 : ((dv1 < dv2) ? -1 : 1);

在doubleValue方法中:

public static double doubleValue(Object value)
        throws NumberFormatException
    {
        if (value == null) return 0.0;
        Class c = value.getClass();
        if (c.getSuperclass() == Number.class) return ((Number) value).doubleValue();
        if (c == Boolean.class) return ((Boolean) value).booleanValue() ? 1 : 0;
        if (c == Character.class) return ((Character) value).charValue();
        String s = stringValue(value, true);
        return (s.length() == 0) ? 0.0 : Double.parseDouble(s);
    }

坑坑坑!!!

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • WPF实现自定义一个自删除的多功能ListBox

    WPF实现自定义一个自删除的多功能ListBox

    这篇文章主要为大家详细介绍了如何利用WPF实现自定义一个自删除的多功能ListBox,文中示例代码讲解详细,感兴趣的小伙伴可以了解一下
    2022-12-12
  • 浅谈Java HttpURLConnection请求方式

    浅谈Java HttpURLConnection请求方式

    这篇文章主要介绍了浅谈Java HttpURLConnection请求方式,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-08-08
  • Java @GlobalLock注解详细分析讲解

    Java @GlobalLock注解详细分析讲解

    这篇文章主要介绍了Java @GlobalLock注解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习吧
    2022-11-11
  • java实现系统托盘示例

    java实现系统托盘示例

    桌面的系统托盘即当程序最小化或者关闭按钮程序并没有退出,而是最小化在任务状态区域,下面是使用java实现系统托盘示例
    2014-03-03
  • Java编程之多线程死锁与线程间通信简单实现代码

    Java编程之多线程死锁与线程间通信简单实现代码

    这篇文章主要介绍了Java编程之多线程死锁与线程间通信简单实现代码,具有一定参考价值,需要的朋友可以了解下。
    2017-10-10
  • MyBatis Plus工具快速入门使用教程

    MyBatis Plus工具快速入门使用教程

    这篇文章主要介绍了MyBatis Plus工具快速入门使用教程,需要的朋友可以参考下
    2018-05-05
  • 全面详解Maven打包及其相关插件和高级特性

    全面详解Maven打包及其相关插件和高级特性

    这篇文章主要为大家介绍了Maven打包及其相关插件和高级特性的全面详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-05-05
  • Java中图片转换为Base64的示例及注意事项

    Java中图片转换为Base64的示例及注意事项

    本文介绍了Base64编码的概念及其作用,同时列举了在实现图片转换为Base64过程中需要注意的问题,包括文件大小、读取异常、图片格式、网络传输效率以及数据安全性等,文中通过代码介绍的非常详细,需要的朋友可以参考下
    2024-10-10
  • PageHelper引发的幽灵数据问题解析

    PageHelper引发的幽灵数据问题解析

    这篇文章主要为大家介绍了PageHelper引发的幽灵数据问题解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-04-04
  • java遍历读取整个redis数据库实例

    java遍历读取整个redis数据库实例

    这篇文章主要介绍了java遍历读取整个redis数据库实例,使用支持正则表达式的key搜索方法jedis.keys(“*”)实现,需要的朋友可以参考下
    2014-05-05

最新评论