动态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=utf82. 创建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); ListqueryBlogIf(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(); } }
需求:从博客表中查询数据,如果不传任何参数,查询所有,传入参数,按照参数查询。
改造我们的BlogMapper.xml
编写测试方法
MyTest.java
public class MyTest { static Logger logger = Logger.getLogger(MyTest.class); @Test public void queryBlogIf(){ SqlSession sqlSession = MyBatisUtils.getSqlSession(); BlogMapper mapper = sqlSession.getMapper(BlogMapper.class); HashMap map = new HashMap(); Listblogs = mapper.queryBlogIf(map); for (Blog blog : blogs) { logger.debug(blog); } sqlSession.close(); } }
发现:当map为空时,查询了所有的数据。
4. 动态SQL常用标签 4.1 where发现:传入参数之后,查询出了满足要求的数据。
在实际的开发中,这种写法是不规范的,不能写1=1的情况。
我们做如下改造即可
4.2 set说明:这个“where”标签会知道如果它包含的标签中有返回值的话,它就插入一个‘where’。此外,如果标签返
回的内容是以AND 或OR 开头的,则它会剔除掉。【这是我们使用的最多的案例】
同理,上面的对于查询 SQL 语句包含 where 关键字,如果在进行更新 *** 作的时候,含有 set 关键词,
BlogMapper.xml
update blog where id = #{id}; title = #{title}, author = #{author}
注意:注意set是用的逗号隔开
测试
public class MyTest { static Logger logger = Logger.getLogger(MyTest.class); @Test public void updateBlog(){ SqlSession sqlSession = MyBatisUtils.getSqlSession(); BlogMapper mapper = sqlSession.getMapper(BlogMapper.class); HashMapmap = new HashMap (); map.put("title","动态SQL"); map.put("author","江南"); map.put("id","7d83bc1ae6a34171b5a1db74c708ea0f"); mapper.updateBlog(map); sqlSession.close(); } }
4.3 choose(when、otherwise)说明:set 元素会动态地在行首插入 SET 关键字,并会删掉额外的逗号
有时候,我们不想用到所有的查询条件,只想选择其中的一个,查询条件有一个满足即可,使用 choose
标签可以解决此类问题,类似于 Java 的 switch 语句
BlogMapper.xml
测试
public class MyTest { static Logger logger = Logger.getLogger(MyTest.class); @Test public void queryBlogChoose(){ SqlSession sqlSession = MyBatisUtils.getSqlSession(); BlogMapper mapper = sqlSession.getMapper(BlogMapper.class); HashMapmap = new HashMap (); map.put("title","动态SQL"); map.put("author","江南"); mapper.queryBlogChoose(map); sqlSession.close(); } }
4.4 Foreach说明:虽然写了好几个条件,但是由于使用了choose,他只会选择其中一个最先符合sql语句的条件进行查询。
动态 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); HashMapmap = 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); HashMap4.5 trimmap = 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(); } }
需求:插入一条博客信息。
相当于这个sql语句:insert into blog (id,title,author,create_time,views) values(5,“redis入门到精通”,“江南雨”,NOW(),1500);,那我们在xml中拼接起来即可,还能判断些值为空值。
之前我们写了一个addBlog的接口用于插入数据,同样的,我们也可以使用insertBlogTrim这个接口来做。
insert into blog values id, title, author, create_time, views #{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); HashMapmap = 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(); } }
5. SQL片段使用了map的方式,更加灵活,在实际开发中更常用。
将多个sql中重复的部分提取出来。增强sql语句的复用性。
title = #{title} and author = #{author}
sql标签和include标签。
6. 小结注意:①、最好基于 单表来定义 sql 片段,提高片段的可重用性
②、在 sql 片段中不要包括 where
其实动态 sql 语句的编写往往就是一个拼接的问题,为了保证拼接准确,我们最好首先要写原生的 sql 语句出来,然后在通过 mybatis 动态sql 对照着改,防止出错。多在实践中使用才是熟练掌握它的技巧。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)