Java神鬼莫测之MyBatis中$与#的区别(五)

简介: Java神鬼莫测之MyBatis中$与#的区别(五)

MyBatis中$与#的区别


1.问题描述


在mybatis的动态传参中, 我们可以私用#{},传递参数,也可以使用${}传递参数, 他们有什么差别呢?


如果在mybatis中的sql语句,使用的是in条件查询,则使用${}时会报错,这又是为什么呢?


2.举例说明


2.1. UserDao接口的方法

//根据字符串usernames,查询所有的User信息
public List<User> findByNames(@Param("usernames") String usernames) throws Exception;

2.2. 调用该方法

public class MybatisTest {
    @Test
    public void test01() throws Exception {
        InputStream is = Resources.getResourceAsStream("SqlMapconfig.xml");
        SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(is);
        SqlSession sqlSession = sqlSessionFactory.openSession();
        UserDao userDao = sqlSession.getMapper(UserDao.class);
        //假设参数是   'lucy','tom'
        String usernames = "'lucy','tom'";
        List<User> list = userDao.findByNames(usernames);
        System.out.println(list);
    }
}

2.3.配置(映射)文件中的配置


2.3.1.使用#{}


① 修改xml设置

<mapper namespace="com.itheima.dao.UserDao">
    <!--根据字符串usernames,查询所有的User信息-->
    <select id="findByNames" resultType="com.itheima.domain.User" >
        select * from user where username in (#{usernames})
    </select>
</mapper>

② 执行删除test01()测试方法


③ 查看执行日志

DEBUG *** ==> Preparing: select * from user where username in (?) 
DEBUG *** ==> Parameters: 'lucy','tom'(String)
DEBUG *** ==> Total: 0

④ 结论


#{} 预编译完成后, 变成了占位符?, 可以预防sql攻击


2.3.2 使用${}


① 修改xml设置

<mapper namespace="com.itheima.dao.UserDao">
    <!--根据字符串usernames,查询所有的User信息-->
    <select id="findByNames" resultType="com.itheima.domain.User" >
        select * from user where username in (${usernames})
    </select>
</mapper>

② 执行删除test01()测试方法


③ 查看执行日志

<mapper namespace="com.itheima.dao.UserDao">
    <!--根据字符串usernames,查询所有的User信息-->
    <select id="findByNames" resultType="com.itheima.domain.User" >
        select * from user where username in (${usernames})
    </select>
</mapper>

② 执行删除test01()测试方法


③ 查看执行日志

DEBUG *** ==> Preparing: Preparing: select * from user where username in ('lucy','tom')
DEBUG *** ==> Parameters: 
DEBUG *** ==> Total: 2
[User{id=1, username='lucy', password='123'}, User{id=2, username='tom', password='123'}]

④ 结论


${} 会单纯的把参数值, 替代到sql语句对应位置上边. 不能预防sql攻击


3.补充


3.1 ${}的场景


  1. 作为in条件时,
  2. 参数为int类型并且数据库中字段的类型是number,
  3. 在sql语句中, 实现动态表名
  4. order by ${},动态排序字段


3.2 ${}的缺点(严重)


${}不能预防sql攻击


例设:


sql语句: select * from ${tablename}


参数值: product; drop product;


那么你的表数据就会被无声无息的干掉了。


目录
相关文章
|
2月前
|
Java 程序员
Java社招面试题:& 和 && 的区别,HR的套路险些让我翻车!
小米,29岁程序员,分享了一次面试经历,详细解析了Java中&和&&的区别及应用场景,展示了扎实的基础知识和良好的应变能力,最终成功获得Offer。
84 14
|
1月前
|
Java
java中面向过程和面向对象区别?
java中面向过程和面向对象区别?
25 1
|
2月前
|
存储 缓存 安全
java 中操作字符串都有哪些类,它们之间有什么区别
Java中操作字符串的类主要有String、StringBuilder和StringBuffer。String是不可变的,每次操作都会生成新对象;StringBuilder和StringBuffer都是可变的,但StringBuilder是非线程安全的,而StringBuffer是线程安全的,因此性能略低。
65 8
|
3月前
|
Java 数据库连接 Maven
mybatis使用一:springboot整合mybatis、mybatis generator,使用逆向工程生成java代码。
这篇文章介绍了如何在Spring Boot项目中整合MyBatis和MyBatis Generator,使用逆向工程来自动生成Java代码,包括实体类、Mapper文件和Example文件,以提高开发效率。
163 2
mybatis使用一:springboot整合mybatis、mybatis generator,使用逆向工程生成java代码。
|
2月前
|
Java
Java代码解释++i和i++的五个主要区别
本文介绍了前缀递增(++i)和后缀递增(i++)的区别。两者在独立语句中无差异,但在赋值表达式中,i++ 返回原值,++i 返回新值;在复杂表达式中计算顺序不同;在循环中虽结果相同但使用方式有别。最后通过 `Counter` 类模拟了两者的内部实现原理。
Java代码解释++i和i++的五个主要区别
|
2月前
|
SQL 安全 Java
MyBatis(6)#{}和${}的区别
在MyBatis中,`#{}`和`${}`是用于在SQL语句中嵌入参数的两种方式。`#{}`用于预处理参数,可以防止SQL注入;而`${}`进行直接字符串替换,适用于动态插入表名或列名,但存在SQL注入风险。建议优先使用`#{}`,并在必要时谨慎使用`${}`。
|
3月前
|
搜索推荐 Java 数据库连接
Java|在 IDEA 里自动生成 MyBatis 模板代码
基于 MyBatis 开发的项目,新增数据库表以后,总是需要编写对应的 Entity、Mapper 和 Service 等等 Class 的代码,这些都是重复的工作,我们可以想一些办法来自动生成这些代码。
43 6
|
3月前
|
Java
通过Java代码解释成员变量(实例变量)和局部变量的区别
本文通过一个Java示例,详细解释了成员变量(实例变量)和局部变量的区别。成员变量属于类的一部分,每个对象有独立的副本;局部变量则在方法或代码块内部声明,作用范围仅限于此。示例代码展示了如何在类中声明和使用这两种变量。
|
3月前
|
Java
Java基础之 JDK8 HashMap 源码分析(中间写出与JDK7的区别)
这篇文章详细分析了Java中HashMap的源码,包括JDK8与JDK7的区别、构造函数、put和get方法的实现,以及位运算法的应用,并讨论了JDK8中的优化,如链表转红黑树的阈值和扩容机制。
43 1
|
3月前
|
Java
Java代码解释静态代理和动态代理的区别
### 静态代理与动态代理简介 **静态代理**:代理类在编译时已确定,目标对象和代理对象都实现同一接口。代理类包含对目标对象的引用,并在调用方法时添加额外操作。 **动态代理**:利用Java反射机制在运行时生成代理类,更加灵活。通过`Proxy`类和`InvocationHandler`接口实现,无需提前知道接口的具体实现细节。 示例代码展示了两种代理方式的实现,静态代理需要手动创建代理对象,而动态代理通过反射机制自动创建。