mybatis的#{}和${}的区别以及order by注入问题

前言略,直奔主题..

#{}相当于jdbc中的preparedstatement

${}是输出变量的值

你可能说不明所以,不要紧我们看2段代码:

String sql = "select * from admin_domain_location order by ?";
PreparedStatement st = con.prepareStatement(sql);
st.setString(1, "domain_id");
System.out.println(st.toString());

ResultSet rs = st.executeQuery();

while(rs.next()){

System.out.println(rs.getString("domain_id"));

}

输出结果:

com.mysql.jdbc.PreparedStatement@1fa1ba1: select * from admin_domain_locationorder by 'domain_id'

3

4

5

2

6

这是个jdbc的preparedstatement例子,不要吐槽我这么写是否合法,这里只是为了说明问题.

以上例子有得出以下信息: 1) order by后面如果采用预编译的形式动态输入参数,那么实际插入的参数是一个字符串,例子中是:order by  'domain_id'

2)输出结果并没有排序,从sql语句中的形式我们也可以推测出此sql语句根本也不合法(正常应该是 order by domain_id )

修改以上代码如下:

String input = "domain_id";
String sql = "select * from admin_domain_location order by "+input;
PreparedStatement st = con.prepareStatement(sql);
System.out.println(st.toString());
ResultSet rs = st.executeQuery();
while(rs.next()){
    System.out.println(rs.getString("domain_id"));
}
输出结果:

com.mysql.jdbc.PreparedStatement@1fa1ba1: select * from admin_domain_locationorder by domain_id

2

3

4

5

6

此次我们直接把一个变量的值拼接sql语句,从结果可以看出来:

1)sql语句拼接正常

2)查询结果排序正常

你可能要问这和#{}与${}有什么关系..

上面已经说过#{}相当于jdbc的preparedstatement,所以以上的第一个例子就相当于#{},那么第二个例子就自然而然指的是${}的情况.

你可能说思维还是有些凌乱,不要紧我们来看第三个例子:

String sql = "select * from admin_domain_location where domain_id=?";
PreparedStatement st = con.prepareStatement(sql);
st.setString(1, "2");
System.out.println(st.toString());
ResultSet rs = st.executeQuery();
while(rs.next()){
  System.out.println(rs.getString("domain_id"));
}
=======================================
String input = "2";
String sql = "select * from admin_domain_location where domain_id='"+input+"'";
PreparedStatement st = con.prepareStatement(sql);
System.out.println(st.toString());
ResultSet rs = st.executeQuery();
while(rs.next()){
  System.out.println(rs.getString("domain_id"));
}
输出结果都为:
com.mysql.jdbc.PreparedStatement@12bf560: select * from admin_domain_location where domain_id='2'
2

这第三个例子虽然说的是#{}和${}通用的问题,也就是说在此种情况下#{}和${}是通用的,只不过需要些小的转换.如例子中需要手动

拼接单引号 ' ' 到变量值的前后,确保sql语句正常.

简单说#{}是经过预编译的,是安全的,而${}是未经过预编译的,仅仅是取变量的值,是非安全的,存在sql注入.

这里先说一下只能${}的情况,从我们前面的例子中也能看出,order by是肯定只能用${}了,用#{}会多个' '导致sql语句失效.此外还有一个like 语句后也需要用${},简单想一下

就能明白.由于${}仅仅是简单的取值,所以以前sql注入的方法适用此处,如果我们order by语句后用了${},那么不做任何处理的时候是存在sql注入危险的.你说怎么防止,那我只

能悲惨的告诉你,你得手动处理过滤一下输入的内容,如判断一下输入的参数的长度是否正常(注入语句一般很长),更精确写查询一下输入的参数是否在预期的参数集合中..


### MyBatis#$区别及实际开发中的选择策略 #### 1. #$的基本概念 在MyBatis中,`#``$`是两种不同的参数替换方式。`#{}`会被MyBatis处理为一个占位符,实际SQL会变为带有`?`的形式,并且参数值会被绑定到占位符上[^4]。这种方式可以有效防止SQL注入攻击。而`${}`则是直接将变量的值插入到SQL语句中,不会进行任何转义或预处理[^1]。 #### 2. 安全性对比 使用`#{}`时,MyBatis会生成PreparedStatement中的占位符`?`,并设置相应的参数值。这种方式是安全的,因为它可以有效防止SQL注入攻击[^5]。例如: ```sql <select id="getUserById" parameterType="int" resultType="User"> SELECT * FROM users WHERE id = #{id} </select> ``` 上述代码中,`#{id}`会被替换为`?`,并且`id`的值会被安全地绑定到SQL语句中。 相比之下,`${}`会直接将变量的值插入到SQL语句中,不经过任何转义处理。这种方式存在SQL注入的风险,因此需要谨慎使用[^2]。例如: ```sql <select id="getUsersByOrder" parameterType="map" resultType="User"> SELECT * FROM users ORDER BY ${orderColumn} </select> ``` 如果`orderColumn`的值是由用户输入的,则可能会导致SQL注入攻击[^3]。 #### 3. 使用场景对比 - **`#{}`的适用场景**:适用于所有类型的参数,包括基本数据类型、字符串、集合、对象等。它适合用于需要传递动态参数但不需要直接拼接SQL的场景。 - **`${}`的适用场景**:适用于需要动态拼接SQL的场景,例如表名、列名或排序字段等。由于其安全性较低,必须确保传入的值是可信的,避免SQL注入风险。 #### 4. 性能对比 从性能角度来看,`#{}``${}`没有本质区别。但由于`#{}`通过PreparedStatement实现,数据库可以缓存执行计划,因此在多次执行相同SQL时,`#{}`可能会稍微优于`${}`[^4]。 #### 5. 实际开发中的选择策略 在实际开发中,应优先使用`#{}`以确保SQL的安全性。只有在确实需要动态拼接SQL(如表名、列名)时才使用`${}`,并且必须对传入的值进行严格的校验过滤[^3]。例如: ```sql <select id="dynamicTableQuery" parameterType="map" resultType="User"> SELECT * FROM ${tableName} WHERE id = #{id} </select> ``` 上述代码中,`tableName`的值必须是经过验证的可信值,而`id`则通过`#{}`安全地绑定到SQL语句中[^2]。 #### 注意事项 - 在使用`${}`时,务必确保传入的值是可控的,避免因用户输入导致的安全问题。 - 对于复杂的SQL查询,可以结合使用`<if>`、`<where>`等动态SQL标签,减少直接拼接SQL的需求[^1]。 ### 示例代码 以下是一个结合`#{}``${}`的示例: ```sql <select id="queryUsers" parameterType="map" resultType="User"> SELECT * FROM ${tableName} <where> <if test="name != null"> AND name = #{name} </if> <if test="age != null"> AND age = #{age} </if> </where> ORDER BY ${orderBy} </select> ```
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值