通过阅读本文你可以更好的理解两个知识点:
1.#{}与${}在实际项目中的使用,避免在项目中使用不当造成不可预知的Bug;
2.MySQL中in里面如果是字符串的话,为什么只取第一个对应的数据,eg: in (“5,6,7”);
------------------------------------------------------------------------------------------------------------
快过年了,项目决定年后上线,在修Bug阶段,在Bug解决后开始当水手划水了,然后复习了下MyBatis的动态SQL的知识,无意中看到一个大佬写的代码,发现和我们平常写的方式有点不同,觉得这样写会产生问题,于是乎开启了思考。正常情况下,我们在用in的时候,参数都用List或者一个Vo对象(里面有一个List),然后用foreach的方式循环取值,但是这个大佬居然in里面是String类型的参数,这样写能查出值吗?
通常的写法:
注:stockInBands是vo对象中的一个List类型的属性。
质疑大佬的写法:
这个String类型的orgIds取值过程如下:
使用了StringUtils.join()的方法将List转为逗号隔开的String类型,我开始了思考,这样的话最后这个String不是变成了类似(“5,6,7”)这样的格式吗?MySQL中in(“5,6,7”)这样的格式可以查出来吗?然后我就开始在数据库中写SQL语句尝试,发现了一个奇怪的现象,这样的格式可以查出来,但是永远只能查出第一条数据,在例子中就是id为5对应的数据,后面的6,7为啥查不出来?
表数据如下:
发现这五条SQL都能查出数据,但是前三条SQL能查出3条数据,而后两条SQL都只能查询出id为5对应的数据。
我开始百度查阅相关知识点,发现在MySQL中in里面如果是字符串的话,会自动使用类似CAST(‘5,6,7‘ AS INT)方法转化成int类型,对于数字开头的字符串来说,转为数字的结果就是截取前面的数字部分(本例中第一个逗号前的数据),对于开头部分不能截取出数字的字符串来说,转换的结果自然就是0了,如下图实例。
这个疑惑解决后,我想了下难道这个大佬写的真有问题,只是他和测试没发现问题吗?我又仔细看了下代码后发现这个大佬写的SQL里面用的是$而不是#,难道用$就能查询正常吗?然后我开始复习#{}与${}的区别,发现果然如此,这个大佬因为用的是$,${} 解析之后是什么就是什么,他不会当做字符串处理。假设传入参数是”Smith”会解析成:order by Smith或者in (Smith),所以并没有出现我发现的那种只会查出第一个id为5对应数据的问题。原来只是因为自己知识浅薄,造成的误解。