为什么会有动态SQL,把SQL写死不是比较方便吗?其实有很多的举例,这里我那一个常见的来说,像我们用户注册,会有必填字段和非必填字段,有些传来的参数不一样,那对应的SQL也不一样,因此,在这些的场景底下我们需要使用动态SQL来完成。
这里动态SQL我是使用XML的办法进行说明,因为使用注解不方便,而且不好观察哪里有误。这里我会先从一些常用的标签的介绍来对动态SQL进行使用的讲解。
1.1 <if>标签
<insert id="insertByCondition">
insert into userinfo(
<if test="username != null">
username
</if>
<if test="password != null">
,password
</if>
<if test="age != null">
,age
</if>
<if test="gender != null">
,gender
</if>
) values (
<if test="username != null">
#{username}
</if>
<if test="password != null">
,#{password}
</if>
<if test="age != null">
,#{age}
</if>
<if test="gender != null">
,#{gender}
</if>
)
</insert>
从上述代码可以简单看出来,<if> 标签是一个<if test = "xxxx">xxxx</if>的结构,然后另一个是运行后的结果,这里发现如果if里面都有值的话,可以完好的插入,但是它就一定没问题吗?下面一个图我就简单注释掉几个来,实验一下。
从上图可以简单的看错来,它是一个BadSql因为我们注掉了username导致,多了一个逗号那怎么样才能避免上述情况呢?那就要引入新的注解。
1.2<trim> 标签
这里我们先简单介绍一下<trim>标签里面几个常见的属性:
1. prefix:表示整个语句块,以prefix的值作为前缀。
2. suffix:表示整个语句块,以suffix的值作为后缀。
3. prefixOverrides:表示整个语句块要去除掉前缀。
4. suffixOverrides:表示整个语句块要去掉后缀。
<insert id="insertByCondition">
insert into userinfo
<trim prefix="(" suffix=")" prefixOverrides=",">
<if test="username != null">
username
</if>
<if test="password != null">
,password
</if>
<if test="gender != null">
,gender
</if>
<if test="age != null">
,age
</if>
</trim>
values
<trim prefix="(" suffix=")" prefixOverrides=",">
<if test="username != null">
#{username}
</if>
<if test="password != null">
,#{password}
</if>
<if test="gender != null">
, #{gender}
</if>
<if test="age != null">
,#{age}
</if>
</trim>
</insert>
上述因为username是不能为空,因此我把注掉username改为注掉gender,可以发现在<if>标签里面的问题给解决了。
1.3 <where>标签
<select id="queryUserByCondition" resultType="com.jincheng.mybatisdemo.model.UserInfo">
select * from userinfo
<where>
<if test="age != null">
age = #{age}
</if>
<if test="gender != null">
and gender = #{gender}
</if>
<if test="deleteFlag">
and delete_flag = #{deleteFlag}
</if>
</where>
</select>
<select id="queryUserByCondition" resultType="com.jincheng.mybatisdemo.model.UserInfo">
select * from userinfo
<trim prefix="where" prefixOverrides="and">
<if test="age != null">
age = #{age}
</if>
<if test="gender != null">
and gender = #{gender}
</if>
<if test="deleteFlag">
and delete_flag = #{deleteFlag}
</if>
</trim>
</select>
通过上面可以发现第一个是使用<where>标签进行写的,而另一个是通过使用<trim>标签进行写的。可以简单比对一下<where>标签,生成where关键字,并且去除最前面的and 或者 or,如果where标签代码块,没有一个查询条件,会省略掉where关键字。也可以使用<trim prefix="where" prefixOverrides="and">替换,但是这种情况下,子元素没有内容时,where关键字也会保留。
1.4 <set>标签
<update id="updateByCondition">
update userinfo
<set>
<if test="password != null">
password = #{password},
</if>
<if test="age != null">
age = #{age},
</if>
<if test="gender != null">
gender = #{gender}
</if>
</set>
where id = #{id}
</update>
这里<set>标签,生成set关键字,去掉整个代码块后面的逗号。如果使用<trim prefix="set" suffixOverrides=",">可以进行替换。
1.5 <foreach>标签
简单来说一下<foreach>标签里面的属性:
1. collection:绑定方法参数中的集合,如LIst,Set,Map或数组对象
2. item:遍历是的每一个对象
3. open:语句块开头的字符串
4. close:语句块结束的字符串
5. separator:每次遍历之间间隔的字符串。
一般用在批量删除中等等。
<delete id="batchDelete">
delete from userinfo
where id in
<foreach collection="ids" open="(" close=")" item="id" separator=",">
#{id}
</foreach>
</delete>
1.6 <inlcude>标签
这个标签用于在xml映射文件中配置的SQL,有时可能会存在很多重复的片段,此时就会在很多多余的代码。
这里我们需要对重复的代码片段进行抽取,将其通过<sql>标签封装到一个SQL片段,然后通过<include>标签进行引用:
1.<sql> : 定义可重用的SQL片段
2. <include>:通过属性refid,指定包含的SQL片段。
<sql id="selectAll">
select * from userinfo
</sql>
<select id="select" resultType="com.jincheng.mybatisdemo.model.UserInfo">
<include refid="selectAll"></include>
</select>