MyBatis中使用$和#所遇到的问题及解决办法

作者:RunforLove 时间:2023-03-25 12:27:55 

在上篇文章给大家介绍了Mybatis中#{}和${}传参的区别及#和$的区别小结,如果大家有需要可以参考下。

$和#简单说明:

#相当于对数据 加上 双引号,$相当于直接显示数据。

一、总结

mybatis中使用sqlMap进行sql查询时,经常需要动态传递参数。动态SQL是mybatis的强大特性之一,也是它优于其他ORM框架的一个重要原因。mybatis在对sql语句进行预编译之前,会对sql进行动态解析,解析为一个BoundSql对象,也是在此处对动态SQL进行处理的。在动态 SQL 解析阶段,#{ }和${ }会有不同的表现,#{ }解析为一个JDBC预编译语句(prepared statement)的参数标记符。

一个 #{ } 被解析为一个参数占位符 ? 。${ } 仅仅为一个纯碎的 string 替换,在动态 SQL 解析阶段将会进行变量替换。

二、Bug描述

前端传入参数:

skip:0
take:10
ruleName:A,B,C

业务层处理:


package SQL;
/**
* 将前端多选参数转义为SQL语句内容
*/
public class SQLUtil {
private final static String REPLACECHAR_COMMA = ",";
private final static String REPLACECHAR_SEMICOLON = ";";
public static void main(String[] args) {
String s1 = "A,B,C";
String s2 = "A B C";
System.out.println("逗号分隔:" + formatInStr(s1));
System.out.println("空格分隔:" + formatInStr(s2));
}
private static String formatInStr(String queryStr) {
return queryInStr(sliptQueryStr(queryStr));
}
private static String[] sliptQueryStr(String queryStr) {
if (null == queryStr || "".equals(queryStr.trim())) return null;
queryStr = queryStr.replaceAll(SQLUtil.REPLACECHAR_COMMA, " ").replaceAll(REPLACECHAR_SEMICOLON, " ");
return queryStr.split("\\s+");
}
private static String queryInStr(String[] queryStrs) {
if (null == queryStrs || 0 == queryStrs.length) return null;
StringBuffer buf = new StringBuffer();
for (int i = 0; i < queryStrs.length; i++) {
if (i != 0) buf.append(",");
buf.append("'").append(queryStrs[i]).append("'");
}
return buf.toString();
}
}

Mapper层处理:


//错误的处理
<if test="ruleName != null and ruleName != ''">
AND a.rule_name IN (#{ruleName})
</if>
//正确的处理
<if test="ruleName != null and ruleName != ''">
AND a.rule_name IN (${ruleName})
</if>

日志描述:


[DEBUG] [2016-08-02 17:42:42.226] [qtp1457334982-157] java.sql.Connection - ==> Preparing: SELECT a.id, a.is_valid, a.rule_lable, a.rule_name, a.type, b.sp_id, b.sp_name, a.rule_content, c.user_name, a.gmt_modified, a.ordering FROM idc_logistics_assign_rules a LEFT JOIN app_user c on c.work_no=a.modifier and c.is_deleted='n', idc_sp_info b WHERE a.is_deleted = 'n' AND b.is_deleted = 'n' AND a.sp_id = b.sp_id AND a.rule_name IN (?) ORDER BY ordering asc limit ?, ?
[DEBUG] [2016-08-02 17:42:42.226] [qtp1457334982-157] java.sql.PreparedStatement - ==> Parameters: 'A','B'(String), 0(Integer), 10(Integer)

结果分析:mapper层对sql有预编译处理,对于#有占位符?,但是对于$会直接替换。

PS:MyBatis排序时使用order by 动态参数时需要注意,用$而不是#

字符串替换

 默认情况下,使用#{}格式的语法会导致MyBatis创建预处理语句属性并以它为背景设置安全的值(比如?)。这样做很安全,很迅速也是首选做法,有时你只是想直接在SQL语句中插入一个不改变的字符串。比如,像ORDER BY,你可以这样来使用:


 ORDER BY ${columnName}

 这里MyBatis不会修改或转义字符串。

重要:接受从用户输出的内容并提供给语句中不变的字符串,这样做是不安全的。这会导致潜在的SQL注入攻击,因此你不应该允许用户输入这些字段,或者通常自行转义并检查。

标签:mybatis,$,#
0
投稿

猜你喜欢

  • Android百度地图应用之MapFragment的使用

    2022-07-07 21:16:37
  • JavaSwing FlowLayout 流式布局的实现

    2023-10-02 03:59:41
  • java操作mongodb之多表联查的实现($lookup)

    2023-08-08 10:24:07
  • 基于c# 接口的实例详解

    2023-10-22 22:22:11
  • Java桥梁设计模式优雅地将抽象与实现分离

    2023-12-11 14:56:36
  • Android标题栏最右边添加按钮的实例

    2021-08-09 07:08:07
  • C#使用FolderBrowserDialog类实现选择打开文件夹方法详解

    2022-08-11 07:28:39
  • 深入浅析JDK8新特性之Lambda表达式

    2023-06-22 08:38:58
  • Java构造函数的相互调用代码示例

    2023-07-01 21:42:11
  • Android仿滴滴出行验证码输入框功能实例代码

    2022-12-08 05:15:49
  • Java使用Jedis操作Redis服务器的实例代码

    2023-09-08 07:38:42
  • java10下编译lombok注解代码分享

    2023-06-06 11:02:35
  • 深入理解java动态代理的两种实现方式(JDK/Cglib)

    2023-11-26 13:29:52
  • 浅谈Spring Boot 整合ActiveMQ的过程

    2022-03-22 05:20:27
  • C# List集合中获取重复值及集合运算详解

    2022-06-13 17:15:45
  • 详解maven中profiles使用实现

    2022-11-13 23:14:24
  • 使用GPS经纬度定位附近地点(某一点范围内查询)

    2023-08-22 18:49:09
  • 新闻列表的分页查询java代码实现

    2022-02-24 19:46:25
  • C#将Excel中的数据转换成DataSet

    2021-10-29 18:20:55
  • c# 插入数据效率测试(mongodb)

    2021-12-16 12:46:07
  • asp之家 软件编程 m.aspxhome.com