[TOC]
mybatis中#和$的区别
#{parameterName}
${parameterName}
首先,我们说一下这两种引用参数时的区别,使用#{parameterName}引用参数的时候,Mybatis会把这个参数认为是一个字符串,并自动加上’’,例如传入参数是“Smith”,那么在下面SQL中:
1 | Select * from emp where name = #{employeeName} |
使用的时候就会转换为:
1 | Select * from emp where name = 'Smith'; |
同时使用${parameterName}的时候在下面SQL中
1 | Select * from emp where name = ${employeeName} |
就会直接转换为:
1 | Select * from emp where name = Smith |
简单说#{}是经过预编译的,是安全的。
而${}是未经过预编译的,仅仅是取变量的值,是非安全的,存在SQL注入。
1 | #{} 这种取值是编译好SQL语句再取值 |
下面我们用一个实际的例子看看分别使用和是否可以防止SQL注入。
首先是使用#{}:
1 | <!-- 使用#{} --> |
分别测试正常传参和拼接传参:
1 | // 使用#{} 正常传参 |
结果如下:
1 | DEBUG [http-nio-8080-exec-5] - ==> Preparing: select * from user where account = ? and password = ? |
很明显,使用#{}的时候,即使传入了恶意参数,#{}只会将其作为一个占位符的参数,如上面这个例子:
1 | DEBUG [http-nio-8080-exec-5] - ==> Preparing: select * from user where account = ? and password = ? |
1 | select * from user where account = '201301001' and password = '111111 or account = 'admin'' |
现在是使用${}:
1 | <!-- 使用${} --> |
分别测试正常传参和拼接传参:
1 | // 使用${} 正常传参 |
结果如下:
1 | DEBUG [http-nio-8080-exec-18] - ==> Preparing: select * from user where account = 201301001 and password = 111111 |
很明显,使用${}将参数拼接后在编译成SQL语句,不能防止SQL注入,查询出了有关account=admin的额外信息,这是很危险的。
返回主键id