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

目录
  • test条件中单引号双引号问题
    • 具体原因
  • 动态sql中test的一些问题
    • mybatis动态sql中OGNL中type=="1"和type='1'的区别
    • 解决方案

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);
    }

坑坑坑!!!

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

(0)

相关推荐

  • mybatis 对于生成的sql语句 自动加上单引号的情况详解

    目录 对于生成的sql语句 自动加上单引号的情况 mySQL中replace的用法 mybatis中IFNULL(P1,P2)函数的用法 mybatis单引号字母逻辑处理的一个坑 原因分析 对于生成的sql语句 自动加上单引号的情况 mybatis是这样的,如果表的字段跟系统字段冲突,写sql语句的时候必须得加上单引号,这样才会区分 mySQL中replace的用法 1.replace into replace into table (id,name) values('1','aa'),('2'

  • Mybatis中单双引号引发的惨案及解决

    目录 #{}与${}的区别 问题 最后 #{}与${}的区别 #{}是预编译处理,${}是字符串替换Mybatis在处理#{}时,会将sql中的#{}替换为?号, 调用PreparedStatement的set方法来赋值: Mybatis在处理时 , 就 是 把 {}时,就是把时,就是把{}替换成变量的值. 使用#{}可以有效的防止SQL注入,提高系统安全性. 再通俗的说,使用${}mybatis会把参数加上双引号,而${} 你给啥,sql语句中就是啥,如下示例: select * from t

  • 解决mybatis #{}无法自动添加引号的错误

    目录 mybatis #{}无法自动添加引号 解决 mybatis #{}与${} 单引号 解决办法 验证 mybatis #{}无法自动添加引号 传入string类型时,无法自动添加引号,导致SQL将值识别为列名,导致SQL失败 解决 使用map类型代替string的传值 如 Map<String, String> map = new HashMap<>(2); map.put("userName", userName); return userMapper.

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

    目录 test条件中单引号双引号问题 具体原因 动态sql中test的一些问题 mybatis动态sql中OGNL中type=="1"和type='1'的区别 解决方案 test条件中单引号双引号问题 在mybatis中test判断条件中使用单引号会报错 通常使用双引号 通常test后的判断条件写在双引号内,但是当条件中判断使用字符串时应该如下方式开发 <when  test="channel ==null" > <when  test='chan

  • 替换php字符串中的单引号为双引号的方法

    实例如下: $param = "{'id':'12', 'name':'hi'}"; $new = preg_replace('/\"/', '"', $param); 以上这篇替换php字符串中的单引号为双引号的方法就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持我们.

  • php中的单引号、双引号和转义字符详解

    PHP单引号及双引号均可以修饰字符串类型的数据,如果修饰的字符串中含有变量(例$name):最大的区别是: 双引号会替换变量的值,而单引号会把它当做字符串输出. 例如: <?php $name="string"; echo " 字符串" . '$name'; echo "字符串" . "$name"; ?> 结果: 字符串$name 字符串string 转义字符,顾名思义会将规定的语法用"\"来

  • 解决myBatis中删除条件的拼接问题

    今天刚刚学习了mybatis,做了简单的对数据库的增删改查.在进行删除操作时,单条删除时很简单,但是批量删除的时候拼接删除条件却有些麻烦,现记录一下做法. Sql语句中,当删除条件并不唯一的时候,我们有两种删除的sql语句,一种使用or拼接where中的条件,例如delete from 表名where 条件1 or 条件2,另一种是使用in 例如delete from 表名where 元素in( ) 利用第一种删除语句在mybatis中的mapping.xml中进行拼接: 利用第二种删除语句在m

  • MyBatis中多条件查询商品的三种方法及区别

    目录 一.Sql语句设置多个参数有几种方式 二.代码附上 一.Sql语句设置多个参数有几种方式 1.散装参数:需要使用@Param标记Sql语句中占位符处的名称例如 #{name} 2.实体类封装参数 只需要保证Sql中的参数名和实体类属性名对应上,即可设置成功BrandMapper.xml中的SQL语句不用动,把TestBrandMapper中的代码修改即可 3.Map集合 只需要保证Sql中的参数名和Map集合的键的名称对应上,即可设置成功.BrandMapper.xml中Sql语句不用改

  • js 单引号替换成双引号,双引号替换成单引号的实现方法

    1.双引号替换成单引号 var _adrobj = JSON.stringify(address).replace(/\"/g,"'");ps: var a = {a:1,b:2}; JSON.stringify(a)====>"{"a":1,"b":2}" 2.单引号替换成双引号 var _nstr = _adrdata.replace(/'/g, '"');ps: var json = JSON.

  • 基于mybatis中<include>标签的作用说明

    MyBatis中sql标签定义SQL片段,include标签引用,可以复用SQL片段 sql标签中id属性对应include标签中的refid属性.通过include标签将sql片段和原sql片段进行拼接成一个完整的sql语句进行执行. <sql id="sqlid"> res_type_id,res_type </sql> <select id="selectbyId" resultType="com.property.vo

  • javascript 复杂的嵌套环境中输出单引号和双引号

    <a href="javascript:alert('这里有2个单引号一个双引号"\'\' :)');" >引号的嵌套</a> 单引号 ' 的 ASCII码是 39 双引号 " 的 ASCII码是34,所以我们可以用&#xxx;的形式来代替具体的符号.虽然变成了&#xxx;的形式,但是在嵌套环境中仍然要很当心.这里代码里的alert()使用的引号是单引号,alert里的字符串应该转义掉单引号.于是就有了 \'的形式.反之,就要

  • JavaScript中的单引号和双引号报错的解决方法

    在使用JavaScript显示消息或者传递字符数据的时候,经常会碰到数据中夹杂单引号(')或者双引号("),这种语句往往会造成JavaScript报错.对此一般采用/'或者/"的解决. 例如: Alert("this is test "message"!"); Alert('this is test 'message'!'); 一般会改成以下语句 Alert("this is test /"message/"!&qu

  • Js参数值中含有单引号或双引号问题的解决方法

    <script type="text/javascript">function Display(LoginEmail, UserName, ID) {        alert(LoginEmail);    }</script> 1.使用单引号传参:<a href="javascript:Display('abc@abc.com','Aleax Xie','10101239393')">Test1</a> 如果其中一

随机推荐