动态 SQL,通过 MyBatis 提供的各种标签对条件作出判断以实现动态拼接 SQL 语句。这里的条件判断使用的表达式为 OGNL 表达式。常用的动态 SQL 标签有< if >、< where >、< choose >、< foreach >等。MyBatis 的动态 SQL 语句,与 JSTL 中的语句非常相似。
动态 SQL,主要用于解决查询条件不确定的情况:在程序运行期间,根据用户提交的查询条件进行查询。提交的查询条件不同,执行的 SQL 语句不同。若将每种可能的情况均逐一列出,对所有条件进行排列组合,将会出现大量的 SQL 语句。此时,可使用动态 SQL 来解决这样的问题
注意: 在 mapper 的动态 SQL 中若出现大于号(>)、小于号(<)、大于等于号(>=),小于等于号(<=)等符号,最好将其转换为实体符号。否则,XML 可能会出现解析出错问题。特别是对于小于号(<),在 XML 中是绝不能出现的, 否则解析 mapper 文件会出错。
实体符号表:
对于该标签的执行,当 test 的值为 true 时,会将其包含的 SQL 片断拼接到其所在的 SQL 语句中。
语法:< if test=”条件” > sql 语句的部分
xml
select * from t_user order by ${column} desc
dao
//获取所有学生记录, column排序的参数
List getAllUser(@Param("column") String column);
测试
a.不传入排序参数
ListuserList = userDao.getAllUser(null); userList.forEach(user -> System.out.println(user));
b.传入排序参数user_name
List3.< where >标签userList = userDao.getAllUser("user_name"); userList.forEach(user -> System.out.println(user));
< if >标签的中存在一个比较麻烦的地方:若 where 后的所有< if>条件均为 false,而 where 后若又没有 1=1 子句,则 SQL 中就会只剩下一个空的 where,SQL出错。
所以,在 where 后,需要添加永为真子句 1=1,以防止这种情况的发生。
但当数据量很大时,会严重影响查询效率。
使用< where >标签,在有查询条件时,可以自动添加上 where 子句;没有查询条件时,不会添加
where 子句。需要注意的是,第一个标签中的 SQL 片断,可以不包含 and。不过,写上 and 也不错,系统会将多出的 and 去掉。但其它中 SQL 片断的 and,必须要求写上。否则 SQL 语句将拼接出错.
xml
dao
ListgetAllUser(@Param("name") String name, @Param("password") String password);
测试
List4.< foreach >标签userList = userDao.getAllUser(null, "123456"); userList.forEach(user -> System.out.println(user));
标签用于实现对于数组与集合的遍历。对其使用,需要注意:
➢ collection 表示要遍历的集合类型, list ,array 等。
➢ open、close、separator 为对遍历内容的 SQL 拼接。
(1) 遍历 List<简单类型>
表达式中的 List 使用 list 表示(不管dao接口中的参数取什么名字, 在mapper xml 文件中只能用list),其大小使用 list.size 表示.
xml
dao
ListgetUserByIdList(List myList);
测试
Listlist = new linkedList<>(); list.add(2); list.add(3); list.add(4); List userList = userDao.getUserByIdList(list); userList.forEach(user -> System.out.println(user));
(2) 遍历 List<对象类型>
dao
ListgetUserByUserList(List myList);
测试
List5.< sql >标签list = new linkedList<>(); list.add(new User(2, null, null)); list.add(new User(3, null, null)); list.add(new User(4, null, null)); List userList = userDao.getUserByUserList(list); userList.forEach(user -> System.out.println(user));
< sql >标签用于定义 SQL 片断,以便其它 SQL 标签复用。而其它标签使用该 SQL 片断,需要使用< include >子标签。该< sql >标签可以定义 SQL 语句中的任何部分,所以< include >子标签可以放在动态 SQL的任何位置.
xml
select * from t_user
dao
User getUserById(Integer id);
测试
User user = userDao.getUserById(2);
System.out.println(user);
关于动态SQL的标签还有很多, 详细情况请见官网文档
https://mybatis.org/mybatis-3/zh/dynamic-sql.html



