MySQL——动态SQL拼接

深藏阁楼爱情的钟 2024-04-01 12:54 562阅读 0赞

一、动态sql拼接

目标

  • 能够使用mybatis的标签实现动态SQL拼接

分析

我们在前边的学习过程中,使用的SQL语句都非常简单。而在实际业务开发中,我们的SQL语句通常是动态拼接而成的,比如:条件搜索功能的SQL语句。

  1. # 提供了一个功能:用户可以在页面上根据username、sex、address进行搜索
  2. # 用户输入的搜索条件:可以是一个条件,也可能是两个、三个
  3. # 只输入一个条件:姓名是"王"
  4. SELECT * FROM USER WHERE username LIKE '%王%'
  5. # 只输入一个条件:性别是“男”
  6. SELECT * FROM USER WHERE sex = '男'
  7. # 输入两个条件:姓名“王”,性别“男”
  8. SELECT * FROM USER WHERE username LIKE '%王%' AND sex = '男'
  9. # 输入三个条件:姓名“王”,性别“男”,地址“北京”
  10. SELECT * FROM USER WHERE username LIKE '%王%' AND sex = '男' AND address LIKE '%北京%';

在Mybatis中,SQL语句是写在映射配置的XML文件中的。Mybatis提供了一些XML的标签,用来实现动态SQL的拼接。

常用的标签有:

  • <if></if>:用来进行判断,相当于Java里的if判断
  • <where></where>:通常和if配合,用来代替SQL语句中的where 1=1
  • <foreach></foreach>:用来遍历一个集合,把集合里的内容拼接到SQL语句中。例如拼接:in (value1, value2, ...)
  • <sql></sql>:用于定义sql片段,达到重复使用的目的

讲解

1. 准备Mybatis环境
  1. 创建java项目,导入jar包;准备JavaBean
  2. 创建映射器接口UserDao
  3. 创建映射配置文件UserDao.xml
  4. 创建全局配置文件SqlMapConfig.xml
  5. 创建日志配置文件log4j.properties
2. <if>标签:
语法介绍
  1. <if test="判断条件,使用OGNL表达式进行判断">
  2. SQL语句内容, 如果判断为true,这里的SQL语句就会进行拼接
  3. </if>
使用示例
  • 根据用户的名称和性别搜索用户信息。把搜索条件放到User对象里,传递给SQL语句
  1. 映射器接口UserDao上加方法

    package com.demo.dao;

    import com.demo.domain.User;

    import java.util.List;

  1. public interface UserDao {
  2. /**
  3. * 根据username和sex搜索用户
  4. * @param user 封装了搜索条件的User对象
  5. * @return 搜索的结果
  6. */
  7. List<User> search1(User user);
  8. }
  1. 映射文件UserDao.xml里配置statement

    <?xml version=”1.0” encoding=”UTF-8” ?>
    <!DOCTYPE mapper

    1. PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
    2. "http://mybatis.org/dtd/mybatis-3-mapper.dtd">




  2. 功能测试,在测试类里加测试方法

    package com.demo;

    import com.demo.dao.UserDao;
    import com.demo.domain.User;
    import org.apache.ibatis.io.Resources;
    import org.apache.ibatis.session.SqlSession;
    import org.apache.ibatis.session.SqlSessionFactory;
    import org.apache.ibatis.session.SqlSessionFactoryBuilder;
    import org.junit.After;
    import org.junit.Before;
    import org.junit.Test;

    import java.io.IOException;
    import java.io.InputStream;
    import java.util.List;

  1. public class SqlTest {
  2. private UserDao userDao;
  3. private SqlSession session;
  4. private InputStream is;
  5. /**
  6. * 要求:根据username和sex搜索用户
  7. * 搜索条件放到user对象里
  8. */
  9. @Test
  10. public void testSearch(){
  11. User user = new User();
  12. // user.setUsername("王");
  13. // user.setSex("男");
  14. List<User> userList = userDao.search1(user);
  15. userList.forEach(System.out::println);
  16. }
  17. @Before
  18. public void init() throws IOException {
  19. //1. 读取全局配置文件
  20. is = Resources.getResourceAsStream("SqlMapConfig.xml");
  21. //2. 得到一个SqlSession对象
  22. SqlSessionFactory factory = new SqlSessionFactoryBuilder().build(is);
  23. session = factory.openSession();
  24. userDao = session.getMapper(UserDao.class);
  25. }
  26. @After
  27. public void destroy() throws IOException {
  28. session.close();
  29. is.close();
  30. }
  31. }
3. <where>标签
语法介绍

在刚刚的练习的SQL语句中,我们写了where 1=1。如果不写的话,SQL语句会出现语法错误。Mybatis提供了一种代替where 1=1的技术:<where></where>标签。

代码示例

把上一章节的实现代码进行优化,使用<where></where>标签代替where 1=1

  1. 映射器UserDao的search1方法:已有,不用修改

    /**

    • 根据username和sex搜索用户
    • @param user 封装了搜索条件的User对象
    • @return 搜索的结果
      */
      List search1(User user);
  2. 在映射文件UserDao.xml里修改SQL语句

  3. 在测试类里进行功能测试:测试方法不需要修改

    @Test
    public void testSearch(){

    1. User user = new User();
    2. // user.setUsername("王");
    3. // user.setSex("男");
    4. List<User> userList = userDao.search1(user);
    5. userList.forEach(System.out::println);

    }

4. <foreach>标签
语法介绍

foreach标签,通常用于循环遍历一个集合,把集合的内容拼接到SQL语句中。例如,我们要根据多个id查询用户信息,SQL语句:

  1. select * from user where id = 1 or id = 2 or id = 3;
  2. select * from user where id in (1, 2, 3);

假如我们传参了id的集合,那么在映射文件中,如何遍历集合拼接SQL语句呢?可以使用foreach标签实现。

  1. <!--
  2. foreach标签:
  3. 属性:
  4. collection:被循环遍历的对象,使用OGNL表达式获取,注意不要加#{}
  5. open:循环之前,拼接的SQL语句的开始部分
  6. item:定义变量名,代表被循环遍历中每个元素,生成的变量名
  7. separator:分隔符
  8. close:循环之后,拼接SQL语句的结束部分
  9. 标签体:
  10. 使用#{OGNL}表达式,获取到被循环遍历对象中的每个元素
  11. -->
  12. <foreach collection="" open="id in(" item="id" separator="," close=")">
  13. #{id}
  14. </foreach>
使用示例
  1. 有搜索条件类QueryVO如下:

    package com.itheima.domain;

    public class QueryVO {

    1. private Integer[] ids;
    2. public Integer[] getIds() {
    3. return ids;
    4. }
    5. public void setIds(Integer[] ids) {
    6. this.ids = ids;
    7. }

    }

  2. 在映射器UserDao里加方法

    /**

    1. * QueryVO里有一个Integer[] ids
    2. * 要求:根据ids查询对应的用户列表
    3. */

    List search2(QueryVO vo);

  3. 在映射文件UserDao.xml里配置statement

    1. <select id="search2" resultType="User">
    2. <!--select * from user where id in(41, 42, 45)-->
    3. select * from user where
    4. <foreach collection="ids" open="id in(" item="id" separator="," close=")">
    5. #{id}
    6. </foreach>
    7. </select>
  4. 功能测试

    @Test

    1. public void testSearch2(){
    2. QueryVO vo = new QueryVO();
    3. vo.setIds(new Integer[]{
    4. 41,42,43,44,45});
    5. List<User> userList = userDao.search2(vo);
    6. userList.forEach(System.out::println);
    7. }
5. <sql>标签

在映射文件中,我们发现有很多SQL片段是重复的,比如:select * from user。Mybatis提供了一个<sql>标签,把重复的SQL片段抽取出来,可以重复使用。

语法介绍

在映射文件中定义SQL片段:

  1. <sql id="唯一标识">sql语句片段</sql>

在映射文件中引用SQL片段:

  1. <include refid="sql片段的id"></include>
使用示例

在查询用户的SQL中,需要重复编写:select * from user。把这部分SQL提取成SQL片段以重复使用

  • 要求:QueryVO里有ids,user对象。根据条件进行搜索
  1. 修改QueryVO,增加成员变量user

    package com.itheima.domain;

    /**

    • @author liuyp
    • @date 2021/09/07
      */
      public class QueryVO {

      private Integer[] ids;
      private User user;

      //get/set方法……
      }

  2. 在映射器UserDao里加方法

    /**

    1. * 动态SQL拼接的综合应用:ifwhereforeach
    2. * 要求:QueryVo里有idsusernamesex值,根据这些值进行搜索
    3. */
    4. List<User> search3(QueryVO vo);
  3. 在映射文件UserDao.xml里配置statement

    select * from user



    and username like “%”#{user.username}”%”


    and sex = #{user.sex}


  4. 在测试类里加测试方法

    @Test

    1. public void testSearch3(){
    2. QueryVO vo = new QueryVO();
    3. vo.setIds(new Integer[]{
    4. 41,42,43,44,45});
    5. // User user = new User();
    6. // user.setUsername("王");
    7. // user.setSex("男");
    8. // vo.setUser(user);
    9. List<User> userList = userDao.search3(vo);
    10. userList.forEach(System.out::println);
    11. }

发表评论

表情:
评论列表 (有 0 条评论,562人围观)

还没有评论,来说两句吧...

相关阅读

    相关 Mybatis动态拼接sql

    Mybatis动态拼接sql 需求:查询某张表时条件不确定,可能有一个,可能有多个,也可能没有条件 <!-- 动态sql --> <select id="