MyBatis之动态SQL

阅读 68

2022-02-02

目录

搭建环境

if

choose、when、otherwise

trim、where、set

sql片段

foreach


动态 SQL 是指在不同的情况下,根据不同的条件生成不同的 SQL 语句

搭建环境

数据库

CREATE TABLE `blog`(
	`id` VARCHAR(50) NOT NULL COMMENT '博客id',
	`title` VARCHAR(100) NOT NULL COMMENT '博客标题',
	`author` VARCHAR(30) NOT NULL COMMENT '博客作者',
	`create_time` DATETIME NOT NULL COMMENT '创作时间',
	`views` INT(30) NOT NULL COMMENT '浏览量'
)ENGINE=INNODB DEFAULT CHARSET=utf8

实体类

@Data
@AllArgsConstructor
@NoArgsConstructor
public class Blog {
    private String id;
    private String title;
    private String author;
    private Date createTime;
    private int views;
}

实体类对应 mapper 和 xml 文件

public interface BlogMapper {
}
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
        PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.zhouyue.dao.BlogMapper">

</mapper>

UUID 工具类

public class IDUtils {
    public static String getId(){
        return UUID.randomUUID().toString().replaceAll("-","");
    }
}

插入数据测试

    @Test
    public void test(){
        SqlSession sqlSession = MyBatisUtils.getSqlSession();
        BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
        mapper.addBlog(new Blog(IDUtils.getId(), "MyBatis学习", "苏福唉", new Date(), 154));
        mapper.addBlog(new Blog(IDUtils.getId(), "Spring学习", "苏福唉", new Date(), 632));
        mapper.addBlog(new Blog(IDUtils.getId(), "SpringMVC学习", "苏福唉", new Date(), 148));
        mapper.addBlog(new Blog(IDUtils.getId(), "SpringBoot学习", "苏福唉", new Date(), 248));
        sqlSession.close();
    }

if

实现下面功能:

如果传入 title ,则查询指定博客,如果传入 author ,则查询该作者的所有博客,如果什么都不传,则查询所有博客

我们可以使用 MyBatis 中的 if 来实现该功能

表格数据

接口:通过 map 来接收参数

    //查询博客
    List<Blog> queryBlogIF(Map map);

 sql 语句:在这里我们使用 if 来实现不同情况下,不同 sql 语句的拼接

    <select id="queryBlogIF" parameterType="map" resultType="blog">
        select * from blog where 1=1
        <if test="title != null">
            and title = #{title}
        </if>
        <if test="author != null">
            and author = #{author}
        </if>
    </select>

测试

@Test
    public void test1(){
        SqlSession sqlSession = MyBatisUtils.getSqlSession();
        BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
        Map map = new HashMap();
//        map.put("title", "SpringMVC学习");
//        map.put("author", "苏福唉");
        List<Blog> blogs = mapper.queryBlogIF(map);
        for (Blog blog :
                blogs) {
            System.out.println(blog);
        }
        sqlSession.close();
    }

不传参数

传入 title

传入 author

传入 title 和 author,我们可以发现 sql 语句将两者都拼接上了

choose、when、otherwise

choose、when、otherwise 语句其实就可以理解为 switch、case、default 语句

用该语句来实现如果传入 title ,则查询指定博客,如果传入 author ,则查询该作者的所有博客,如果什么都不传,则查询浏览量为632的博客

sql 语句

<select id="queryBlogIF" parameterType="map" resultType="blog">
        select * from blog where 1=1
        <choose>
            <when test="title != null">
                and title = #{title}
            </when>
            <when test="author != null">
                and author = #{author}
            </when>
            <otherwise>
                and views = 632
            </otherwise>
        </choose>
    </select>

 测试语句和之前一样

不传参数

传入 title

传入 author

传入 title 和 author,我们可以发现 sql 语句只拼接了 title ,即靠前的 when 标签中的内容

 添加一个同名的博客,可以发现两个博客都查出来了

trim、where、set

where

在前面的案例中,为了满足对 where 的拼接我们使用了 1=1 的形式,但是这样显然是不好的,因此我们可以使用 where 标签来进行拼接

where 元素只会在子元素返回任何内容的情况下才插入 “WHERE” 子句。而且,若子句的开头为 “AND” 或 “OR”,where 元素也会将它们去除。

使用 where 来修改上面案例

sql 语句

<select id="queryBlogIF" parameterType="map" resultType="blog">
        select * from blog
        <where>
            <choose>
                <when test="title != null">
                    title = #{title}
                </when>
                <when test="author != null">
                    and author = #{author}
                </when>
                <otherwise>
                    and views = 632
                </otherwise>
            </choose>
        </where>
    </select>

set

set 语句和 where 的作用类似,用于动态更新语句的类似解决方案叫做 setset 元素可以用于动态包含需要更新的列,忽略其它不更新的列,set 元素会动态地在行首插入 SET 关键字,并会删掉额外的逗号(这些逗号是在使用条件语句给列赋值时引入的)。

示例

接口

    //更新博客
    int updateBlog(Map map);

 sql 语句

<update id="updateBlog" parameterType="map">
        update blog
        <set>
            <if test="title != null">
                title = #{title},
            </if>
            <if test="author != null">
                author = #{author}
            </if>
        </set>
        where id = #{id}
    </update>

 测试

@Test
    public void test2(){
        SqlSession sqlSession = MyBatisUtils.getSqlSession();
        BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
        Map map = new HashMap();
        map.put("title", "SpringMVC");
        map.put("author", "苏福哥");
        map.put("id", "a37f63f48d384f069f062f5c688114e5");
        int i = mapper.updateBlog(map);
        System.out.println(i);
        sqlSession.close();
    }

trim

我们可以使用 trim 来实现 where 和 set 的功能

实现 where

    <select id="queryBlogIF" parameterType="map" resultType="blog">
        select * from blog
        <trim prefix="WHERE" prefixOverrides="AND |OR">
            <if test="title != null">
                title = #{title}
            </if>
            <if test="author != null">
                and author = #{author}
            </if>
        </trim>
    </select>

 实现 set

<update id="updateBlog" parameterType="map">
        update blog
        <trim prefix="SET" suffixOverrides=",">
            <if test="title != null">
                title = #{title},
            </if>
            <if test="author != null">
                author = #{author}
            </if>
        </trim>
        where id = #{id}
    </update>

 总的来说,动态 sql 本质上还是 sql 语句,只是我们可以在 sql 层面来执行一个逻辑代码

sql片段

有的时候,我们会将 sql 中一些公共的部分提取出来方便复用,减少代码冗余。例如,在上面的案例中我们写了下面代码

我们可以考虑将其抽取出来,使用 sql 标签

<sql id="someSQL">
        <if test="title != null">
            title = #{title}
        </if>
        <if test="author != null">
            and author = #{author}
        </if>
    </sql>

 再引入该片段,使用 include 标签

同样可以实现对应功能

foreach

我们有如下信息

需要查询作者为苏福唉并且博客 id 为 2 3的博客信息

使用 sql 语句

select * from blog where author = '苏福唉' and (id = 2 or id = 3)

在 MyBatis 中我们就可以使用 foreach 来实现

collection:表示传入的集合名称

item:集合中每一个元素名称

open:以什么开始

close:以什么结尾

separator:分割符

<select id="queryBlogForeach" parameterType="map" resultType="blog">
        select * from blog
        <where>
            author = '苏福唉'
            <!--select * from blog where author = '苏福唉' and (id = 2 or id = 3)-->
            <foreach collection="ids" item="id" open="and (" close=")" separator="or">
                id = #{id}
            </foreach>
        </where>
    </select>

测试:

    @Test
    public void test3(){
        SqlSession sqlSession = MyBatisUtils.getSqlSession();
        BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
        List<Integer> ids = new ArrayList<>();
        ids.add(2);
        ids.add(3);
        Map map = new HashMap();
        map.put("ids",ids);
        List<Blog> blogs = mapper.queryBlogForeach(map);
        for (Blog blog : blogs) {
            System.out.println(blog);
        }
        sqlSession.close();
    }

可以发现成功拼接出了 sql,得到正确结果

总而言之,动态sql其实就是在拼接sql语句  

精彩评论(0)

0 0 举报