• 中文
    • English
  • 注册
  • 查看作者
    • Mybatis:动态SQL

      一.  使用动态SQL进行批量删除

      1.  首先给出对应的接口:

      public interface BookMapper {
            int doDeleteBatch(Integer[] bookIds);
      
      }

      2.  配置mybatis-config.xml,配置内容可以参照一《Mybatis:增删改一文中的第四节。

      3.  接下编写对应的Mapper.xml,其中:

      • collection:内容为我们要遍历的数据类型,数组:array ,List:list,Set:set

      • item:遍历的集合中的每一个对象

      • open:循环开始之前输出的内容

      • close:循环结束之后输出的内容

      • separator:循环的分隔符

      <?xml version="1.0" encoding="UTF-8" ?>
      <!DOCTYPE mapper
              PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
              "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
      <mapper namespace="io.zhangjia.mapper.BookMapper">
          <delete id="doDeleteBatch" parameterType="int">
              DELETE  FROM BOOK
              WHERE BOOK_ID IN
              <foreach item="bookId" collection="array"
                       open="(" separator="," close=")">
                  #{bookId}
              </foreach>
          </delete>
      </mapper>

      值得注意的是,forache中的#{bookId}需要和item中的值名称相同,否则会报错。

      3.  接下来编写测试类即可:

      package io.zhangjia.util;
      
      import io.zhangjia.mapper.BookMapper;
      import io.zhangjia.entity.Book;
      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 java.io.IOException;
      import java.io.InputStream;
      import java.util.List;
      
      public class Main {
          public static void main(String[] args) throws IOException {
              String resource = "mybatis-config.xml";
              InputStream inputStream = Resources.getResourceAsStream(resource);
              SqlSessionFactory build = new SqlSessionFactoryBuilder().build(inputStream);
              SqlSession sqlSession = build.openSession(true);
              BookMapper mapper = sqlSession.getMapper(BookMapper.class);
      
              Integer[] bookIds = {1,2,3,27};
              int i = mapper.doDeleteBatch(bookIds);
      
              sqlSession.close();
              inputStream.close();
          }
      }

      二.  使用动态SQL进行批量添加

      1.  首先给出对应的接口:

      public interface BookMapper {
            int doInsertBatch(List<Book> books);
      }

      2.  配置mybatis-config.xml,配置内容可以参照一《Mybatis:增删改一文中的第四节。

      3.  接下编写对应的Mapper.xml,其中:

      <?xml version="1.0" encoding="UTF-8" ?>
      <!DOCTYPE mapper
              PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
              "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
      <mapper namespace="io.zhangjia.mapper.BookMapper">
       
          <insert id="doInsertBatch" parameterType="book" >
              INSERT INTO BOOK( NAME, AUTHOR, PRICE) VALUE
              <foreach item="book"  collection="list"  separator=",">
                  (#{book.name}, #{book.author}, #{book.price})
              </foreach>
          </insert>
      </mapper>

      值得注意的是,forache中的#{bookId}需要和item中的值名称相同,否则会报错。

      3.  接下来编写测试类即可:

      package io.zhangjia.util;
      
      import io.zhangjia.entity.Book;
      import io.zhangjia.mapper.BookMapper;
      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 java.io.IOException;
      import java.io.InputStream;
      import java.util.ArrayList;
      import java.util.List;
      
      public class Main {
          public static void main(String[] args) throws IOException {
              String resource = "mybatis-config.xml";
              InputStream inputStream = Resources.getResourceAsStream(resource);
              SqlSessionFactory build = new SqlSessionFactoryBuilder().build(inputStream);
              SqlSession sqlSession = build.openSession(true);
              BookMapper mapper = sqlSession.getMapper(BookMapper.class);
      
              List<Book> books = new ArrayList<>();
              for (int i = 0; i < 5; i++) {
                  Book book = new Book("书名"+i,"张甲"+i,i*10d);
                  books.add(book);
      
              }
      
              int i = mapper.doInsertBatch(books);
              System.out.println("成功插入 " + i + "条");
      
              sqlSession.close();
              inputStream.close();
          }
      }

      最后执行的sql语句为:

      INSERT INTO BOOK( NAME, AUTHOR, PRICE) VALUE (?, ?, ?) , (?, ?, ?) , (?, ?, ?) , (?, ?, ?) , (?, ?, ?)

      三.  使用动态SQL进行任意条件查询

      1.  首先说一下根据任意查询要实现的功能:

      • 可以根据Book类的任意字段查询数据

      • 比如给出作者,则可以查询所有同一个作者的所有书籍

      • 比如给出作者,书名,则可以查询作者和书名为指定内容的书籍

      接下来给出对应的接口:

      public interface BookMapper {
             List<Book> query(Book book);
      }

      2.  配置mybatis-config.xml,配置内容可以参照一《Mybatis:增删改一文中的第四节。

      3.  接下编写对应的Mapper.xml,一共有两种实现方法,先来看第一种:

      <?xml version="1.0" encoding="UTF-8" ?>
      <!DOCTYPE mapper
              PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
              "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
      <mapper namespace="io.zhangjia.mapper.BookMapper">
         
          <!--方法一-->
      
          <select id="query" parameterType="book" resultType="book">
              SELECT * FROM BOOK
              WHERE 1 = 1
              <if test="bookId != null">
                  AND BOOK_ID = #{bookId}
              </if>
              <if test="name != null">
                  AND NAME LIKE CONCAT('%',#{name},'%')
              </if>
              <if test="author != null">
                  AND AUTHOR = #{author}
              </if>
              <if test="price != null">
                  AND PRICE = #{price}
              </if>
          </select>
      
      </mapper>

      其中 WHERE 1 = 1 ,是为了避免因AND而引发的错误,假如这里我们没有写1 = 1,而是直接WHERE后面就跟了下面的if语句,那么无论哪个if语句被执行,最后的SQL语句都是:

      SELECT * FROM BOOk WHERE AND......

      多出来的AND便会引发异常,所以为了解决这个问题,我们可以在WHERE后面加入一个恒等式1 = 1,这样

      无论哪个if语句被执行,最后的SQL语句都是:

      SELECT * FROM BOOk WHERE 1 = 1 AND......

      问题便得到了解决。这是第一种解决方案,Mybatis也考虑到了这个问题,所以我们可以使用Mybatis提供第二种解决方案,添加where 元素:

      <?xml version="1.0" encoding="UTF-8" ?>
      <!DOCTYPE mapper
              PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
              "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
      <mapper namespace="io.zhangjia.mapper.BookMapper">
          
          <!--方法二-->
      
          <select id="query" parameterType="book" resultType="book">
              SELECT * FROM BOOk
              <where>
                  <if test="bookId != null">
                      AND BOOK_ID = #{bookId}
                  </if>
                  <if test="name != null">
                      AND NAME LIKE CONCAT('%',#{name},'%')
                  </if>
                  <if test="author != null">
                      AND AUTHOR = #{author}
                  </if>
                  <if test="price != null">
                      AND PRICE = #{price}
                  </if>
              </where>
          </select>
      
      </mapper>

      where 元素只会在至少有一个子元素的条件返回 SQL 子句的情况下才去插入“WHERE”子句。而且,若语句的开头为“AND”或“OR”,where 元素也会将它们去除。

      3.  接下来编写测试类即可:

      package io.zhangjia.util;
      
      import io.zhangjia.entity.Book;
      import io.zhangjia.mapper.BookMapper;
      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 java.io.IOException;
      import java.io.InputStream;
      import java.util.ArrayList;
      import java.util.List;
      
      public class Main {
          public static void main(String[] args) throws IOException {
              String resource = "mybatis-config.xml";
              InputStream inputStream = Resources.getResourceAsStream(resource);
              SqlSessionFactory build = new SqlSessionFactoryBuilder().build(inputStream);
              SqlSession sqlSession = build.openSession(true);
              BookMapper mapper = sqlSession.getMapper(BookMapper.class);
      
              Book book = new Book();
              book.setName("书名0");
              book.setAuthor("张甲0");
      
              List<Book> books = mapper.query(book);
              for (Book b: books) {
                  System.out.println("b = " + b);
              }
      
              sqlSession.close();
              inputStream.close();
          }
      }

      上面的测试类执行的sql语句为:

      SELECT * FROM BOOk WHERE NAME LIKE CONCAT('%',?,'%') AND AUTHOR = ?

      四.  使用动态SQL进行任意值更新

      1.  首先说一下根据任意值更新要实现的功能:

      • 比如传进来一个book对象,如果book对象只有作者,则只更新作者

      • 如果值只有书名,则只更新书名,其他的值均为原值,不修改

      接下来给出对应的接口:

      public interface BookMapper {
          int doUpdate(Book book);
      }

      2.  配置mybatis-config.xml,配置内容可以参照一《Mybatis:增删改一文中的第四节。

      3.  接下编写对应的Mapper.xml,其中:

      <?xml version="1.0" encoding="UTF-8" ?>
      <!DOCTYPE mapper
              PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
              "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
      <mapper namespace="io.zhangjia.mapper.BookMapper">
        
          <update id="doUpdate" parameterType="book">
              UPDATE BOOK
              <set>
      
                  <if test="name != null">
                     NAME = #{name},
                  </if>
                  <if test="author != null">
                      AUTHOR = #{author},
                  </if>
                  <if test="price != null">
                      PRICE = #{price}
                  </if>
              </set>
              WHERE BOOK_ID = #{bookId}
          </update>
      
      
      </mapper>

      注意,不要遗漏</if>前面的逗号(最后一个if不用)。类似的用于动态更新语句的解决方案叫做 set。set 元素可以用于动态包含需要更新的列,而舍去其它的,set 元素会动态前置 SET 关键字,同时也会删掉无关的逗号

      3.  接下来编写测试类即可:

      package io.zhangjia.util;
      
      import io.zhangjia.entity.Book;
      import io.zhangjia.mapper.BookMapper;
      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 java.io.IOException;
      import java.io.InputStream;
      import java.util.ArrayList;
      import java.util.List;
      
      public class Main {
         public static void main(String[] args) throws IOException {
             String resource = "mybatis-config.xml";
             InputStream inputStream = Resources.getResourceAsStream(resource);
             SqlSessionFactory build = new SqlSessionFactoryBuilder().build(inputStream);
             SqlSession sqlSession = build.openSession(true);
             BookMapper mapper = sqlSession.getMapper(BookMapper.class);
      
             Book book = new Book();
             book.setName("书名1");
             book.setBookId(43);
             mapper.doUpdate(book);
      
      
             sqlSession.close();
             inputStream.close();
         }
      }

    • 0
    • 0
    • 0
    • 5.5k
    • 请登录之后再进行评论

      登录

      赞助本站

      • 支付宝
      • 微信
      • QQ

      感谢一直支持本站的所有人!

      单栏布局 侧栏位置: