栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 软件开发 > 后端开发 > Java

mybatis学习笔记(九)。动态SQL,if,where,set,choose(when、otherwise),Foreach,SQL片段讲解。

Java 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

mybatis学习笔记(九)。动态SQL,if,where,set,choose(when、otherwise),Foreach,SQL片段讲解。

文章目录

动态SQL1. 数据库设计2. 创建Mybatis基础工程3. 动态SQL之if4. 动态SQL常用标签

4.1 where4.2 set4.3 choose(when、otherwise)4.4 Foreach4.5 trim 5. SQL片段6. 小结

动态SQL

官方文档:

https://mybatis.net.cn/dynamic-sql.html

什么是动态SQL:动态SQL指的是根据不同的查询条件 , 生成不同的Sql语句。

我们之前写的 SQL 语句都比较简单,如果有比较复杂的业务,我们需要写复杂的 SQL 语句,往往需要拼接,而拼接 SQL ,稍微不注意,由于引号,空格等缺失可能都会导致错误。

那么怎么去解决这个问题呢?这就要使用 mybatis 动态SQL,通过 if, choose, when, otherwise,trim, where, set, foreach等标签,可组合成非常灵活的SQL语句,从而在提高 SQL 语句的准确性的同时,也大大提高了开发人员的效率。

1. 数据库设计

新建一个数据库表:blog
字段:id,title,author,create_time,views

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
2. 创建Mybatis基础工程

创建实体类

Blog.java

@Data
@AllArgsConstructor
@NoArgsConstructor
public class Blog {
    private String id;
    private String title;
    private String author;
    private Date createTime;  // 和数据库字段不一致,可以开启驼峰命名转换
    private int views;
}

在核心配置文件中开启驼峰命名转换,使用导包的方式扫描


    


    
    

创建工具类

IDUtils.java

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

uuid生成不唯一的字符串,作为主键使用。

编写插入数据接口

BlogMapper.java

public interface BlogMapper {
    int addBlog(Blog blog);
    List queryBlogIf(Map map);
    int updateBlog(Map map);
    List queryBlogChoose(Map map);
    List queryBlogForeach(Map map);
    int insertBlogTrim(Map map);
}

编写xml文件





    
        insert into blog (id,title,author,create_time,views)
        values (#{id} ,#{title} ,#{author} ,#{createTime},#{views});
    

编写测试方法

MyTest.java

public class MyTest {
    static Logger logger = Logger.getLogger(MyTest.class);
    @Test
    public void addBlog(){
        SqlSession sqlSession = MyBatisUtils.getSqlSession();
        BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
        Blog blog = new Blog();
        blog.setId(IDUtils.getId());
        blog.setTitle("Mybatis如此简单");
        blog.setAuthor("江南雨");
        blog.setCreateTime(new Date());
        blog.setViews(9999);
        mapper.addBlog(blog);
        blog.setId(IDUtils.getId());
        blog.setTitle("Java如此简单");
        mapper.addBlog(blog);
        blog.setId(IDUtils.getId());
        blog.setTitle("Spring如此简单");
        blog.setViews(1000);
        mapper.addBlog(blog);
        blog.setId(IDUtils.getId());
        blog.setTitle("微服务如此简单");
        blog.setViews(2000);
        mapper.addBlog(blog);
        sqlSession.close();
    }
}


3. 动态SQL之if

需求:从博客表中查询数据,如果不传任何参数,查询所有,传入参数,按照参数查询。

改造我们的BlogMapper.xml





    
    select * from blog
    where 1 = 1
    
        and title = #{title}
    
    
        and author = #{author}
    

我们做如下改造即可


    select * from blog
    
        
            
                title = #{title}
            
            
                and author = #{author}
            
            
                and views = #{views}
            
        
    

测试

public class MyTest {
    static Logger logger = Logger.getLogger(MyTest.class);
    @Test
    public void queryBlogChoose(){
        SqlSession sqlSession = MyBatisUtils.getSqlSession();
        BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
        HashMap map = new HashMap();
        map.put("title","动态SQL");
        map.put("author","江南");
        mapper.queryBlogChoose(map);
        sqlSession.close();
    }
}

说明:虽然写了好几个条件,但是由于使用了choose,他只会选择其中一个最先符合sql语句的条件进行查询。

4.4 Foreach

动态 SQL 的另一个常见使用场景是对集合进行遍历(尤其是在构建 IN 条件语句的时候)。

foreach 元素的功能非常强大,它允许你指定一个集合,声明可以在元素体内使用的集合项(item)和索引(index)变量。它也允许你指定开头与结尾的字符串以及集合项迭代之间的分隔符。这个元素也不会错误地添加多余的分隔符,看它多智能!

为了演示效果,先将数据库中前三个数据的id修改为1,2,3,4;

需求:我们需要查询 blog 表中 id 分别为1,2,3的博客信息。

方式一:

思考:需求相当于写了这个sql,select * from blog where id in(1,2,3);那我们将这个sql在xml中拼接出来即可。

BlogMapper.xml





    
        select * from blog
        
            id in
            
                #{id}
            
        
    

说明:collection:接收集合。item:每次遍历得到的参数。open:以什么开始。close:以什么结束。separator:分隔符。

测试

public class MyTest {
    static Logger logger = Logger.getLogger(MyTest.class);
    @Test
    public void queryBlogForeach(){
        SqlSession sqlSession = MyBatisUtils.getSqlSession();
        BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
        HashMap map = new HashMap();
        ArrayList list = new ArrayList();
        list.add(1);
        list.add(2);
        list.add(3);
        map.put("ids",list);
        List blogs = mapper.queryBlogForeach(map);
        for (Blog blog : blogs) {
            logger.debug(blog);
        }
        sqlSession.close();
    }
}

注意:这里的map中放入的是一个集合,使用list。

方式二:

思考:需求相当于写了这个sql,select * from blog where (id = 1 or id = 2 or id = 3);那我们将这个sql在xml中拼接出来即可。

BlogMapper.xml


    select * from blog
    
        
            id = #{id}
        
    

测试

public class MyTest {
    static Logger logger = Logger.getLogger(MyTest.class);
    @Test
    public void queryBlogForeach(){
        SqlSession sqlSession = MyBatisUtils.getSqlSession();
        BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
        HashMap map = new HashMap();
        ArrayList list = new ArrayList();
        list.add(1);
        list.add(2);
        list.add(3);
        map.put("ids",list);
        List blogs = mapper.queryBlogForeach(map);
        for (Blog blog : blogs) {
            logger.debug(blog);
        }
        sqlSession.close();
    }
}

4.5 trim

需求:插入一条博客信息。

相当于这个sql语句:insert into blog (id,title,author,create_time,views) values(5,“redis入门到精通”,“江南雨”,NOW(),1500);,那我们在xml中拼接起来即可,还能判断些值为空值。

之前我们写了一个addBlog的接口用于插入数据,同样的,我们也可以使用insertBlogTrim这个接口来做。


    insert into blog
    
        id,
        title,
        author,
        create_time,
        views
    
    values
    
         #{id},
         #{title},
         #{author},
         #{createTime},
         #{views}
    

说明:这个sql更体现了动态sql。prefix:以什么开始,suffix:以什么结束,suffixOverrides:以什么分隔。

测试:

public class MyTest {
    static Logger logger = Logger.getLogger(MyTest.class);
    @Test
    public void insertBlogTrim(){
        SqlSession sqlSession = MyBatisUtils.getSqlSession();
        BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
        HashMap map = new HashMap();
        map.put("id",IDUtils.getId());
        map.put("title","redis入门到精通");
        map.put("author","江南雨");
        map.put("createTime",new Date());
        map.put("views",1500);
        mapper.insertBlogTrim(map);
        sqlSession.close();
    }
}

使用了map的方式,更加灵活,在实际开发中更常用。

5. SQL片段

将多个sql中重复的部分提取出来。增强sql语句的复用性。


    
        
            title = #{title}
        
        
            and author = #{author}
        
    
    

sql标签和include标签。

注意:①、最好基于 单表来定义 sql 片段,提高片段的可重用性
②、在 sql 片段中不要包括 where

6. 小结

其实动态 sql 语句的编写往往就是一个拼接的问题,为了保证拼接准确,我们最好首先要写原生的 sql 语句出来,然后在通过 mybatis 动态sql 对照着改,防止出错。多在实践中使用才是熟练掌握它的技巧。

转载请注明:文章转载自 www.mshxw.com
本文地址:https://www.mshxw.com/it/723502.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号