目录
前言
if
trim
where
set
foreach
动态 SQL 是 MyBatis 的强大特性之一。如果你使用过 JDBC 或其它类似的框架,你应该能理解根据不同条件拼接 SQL 语句有多痛苦,例如拼接时要确保不能忘记添加必要的空格,还要注意去掉列表最后一个列名的逗号。利用动态 SQL,可以彻底摆脱这种痛苦。
例如在注册用户的时候, 提交的表单中肯定有非必填字段, 前端会将拿到的数据送给后端, 但是, 用户在提交的时候, 由于存在非必填字段, 于是用户提交的字段可能是变数, 可能是一个, 两个或者三个, 例如有下面三个字段
- username
- password
- tele
- age(非必填)
- sex
- .....
除了age都为必填, 这个时候, 前端返回数据给后端, 然后使用sql将其存入数据库, 那么, 在插入的时候
insert into table(username, password, tele, sex, age) values(#{username},#{username}, #{tele}, #{sex}, #{age});
上面是age传入的时候, 那么试想一下, 如果这个age如果用户没有选择填写呢, 那么insert语句中的age选项就应该被抹除
使用动态 SQL 并非一件易事,但借助可用于任何 SQL 映射语句中的强大的动态 SQL 语言,MyBatis 显著地提升了这一特性的易用性。
如果你之前用过 JSTL 或任何基于类 XML 语言的文本处理器,你对动态 SQL 元素可能会感觉似曾相识。在 MyBatis 之前的版本中,需要花时间了解大量的元素。借助功能强大的基于 OGNL 的表达式,MyBatis 3 替换了之前的大部分元素,大大精简了元素种类,现在要学习的元素种类比原来的一半还要少。
insert into user(
username,
password,
nickname,
sex,
birthday,
head
) values (
#{username},
#{password},
#{nickname},
#{sex},
#{birthday},
#{head}
)
if中有一个test属性, test是判断这个sex项是否存在的标准, 前端传过来的数据会被解析为java对象, 并将属性写入到对象中, 如果对象中的sex属性为空, 那么就代表用户没有传入这个字段, 于是就需要将这个项给抹除.
上面的情况是只有一个选填项, 但是如果所有的字段, 都可能是选填项, 那么就可以配合trim使用if来解决:
insert into user
username,
password,
nickname,
sex,
birthday,
head,
create_time,
#{username},
#{password},
#{nickname},
#{sex},
#{birthday},
#{head},
#{createTime},
head,
create_time,
上面语句翻译过来就是
isnert into user(username, password, age, sex, ) values(#{username}, #{password}, #{age}, #{sex},)
需要注意的是, 这里的sex可能为选填, 无论如何, 这个table_name后面的参数和values中的参数, 必定有以 ',' 结尾的内容, 但是这个在sql中是错误语句. 于是使用suffixOverrides将后缀','去除(不包括 suffix =')').
传⼊的⽤户对象,根据属性做 where 条件查询,⽤户对象中属性不为 null 的,都为查询条件。如user.username 为 "a",则查询条件为 where username="a":
UserMapper 接⼝中新增条件查询⽅法:
List selectByCondition(User user);
xml中新增查询语句:
注意: 这里的and一定要写到 column = #{column } 的前面, 也就是:
' and column = #{column } '
where标签就会自动识别并去除多余的and.
当然, where也可以使用trim标签和if标签代替.
根据传⼊的⽤户对象属性来更新⽤户数据,可以使⽤
定义mapper接口:
int updateById(User user);
xml:
update user
username=#{username},
password=#{password},
nickname=#{nickname},
sex=#{sex},
birthday=#{birthday},
head=#{head},
create_time=#{createTime},
where id=#{id}
set会自动去掉末尾的 ','
对集合进⾏遍历时可以使⽤该标签。
示例:根据多个⽂章 id 来删除⽂章数据
ArticleMapper 中新增接⼝⽅法
int deleteByIds(List ids);
使用List接收, Integer为泛型, 传入的是一个Integer的list集合, 每一个Integer对象对应一个要被删除的文章(Article)
ArticleMapper.xml 中新增删除 sql:
delete from article where id in
#{item}
in后面其实是可以使用一个多表查询的, 现在这里使用的foreach进行遍历.
然后就遍历成了:
delete from article where id in (item1, item2, item3 ..... )
这一个个item对象就是list中的Integer对象. 也就是对应的int类型, 与id的类型对应.