• MyBatis之动态SQL&&分页


    目录

    一、MyBatis动态SQL

            1.1 mybatis的foreach标签

    二、模糊查询

    三、mybatis对结果集的处理

    四、分页查询

            4.1 第三方分页插件

            4.2 集成mybatis进行使用

    五、对特殊字符的处理


    一、MyBatis动态SQL

            1.1 mybatis的foreach标签

            我们一起要实现mybatis的动态SQL,就要了解他的两个标签:if、forEach

            我们利用in关键词查询的案例来一起看看关于forEach标签的使用;

            首先就是在昨天Leaf发布的文章:MyBatis环境搭建&&测试增删改查__Leaf1217的博客-CSDN博客

    里面介绍的插件自动生成的一个配置文件里配置我们要添加的方法配置信息:BookMapper.xml

    这里的配置很重要,着重看看forEach标签以及它的属性。 

    1. <select id="selectByIn" resultMap="BaseResultMap" parameterType="java.util.List" >
    2. select
    3. <include refid="Base_Column_List" />
    4. from t_mvc_book
    5. where bid in
    6. <foreach collection="bookIds" open="(" close=")" separator="," item="bid">
    7. #{bid}
    8. foreach>
    9. select>

    然后在自动生成的方法接口BookMapper.java里面也添加上方法:selectByIn

    1. //通过in关键字进行查询,讲解foreach标签的使用
    2. //如果参数 是 非实体类(book、order、...),那么记得加上注解 @param
    3. //bookIds是对应collection属性的
    4. List selectByIn(@Param("bookIds") List bidIds);

    然后再在我们自己写的实际使用的接口中也添加上:selectByIn

    1. package com.leaf.biz;
    2. import com.leaf.model.Book;
    3. import org.apache.ibatis.annotations.Param;
    4. import java.util.List;
    5. /**
    6. * @author Leaf
    7. * @site 2977819715
    8. * @company 玉渊工作室
    9. * @create  2022-08-11 1:09
    10. */
    11. public interface BookBiz {
    12. int deleteByPrimaryKey(Integer bid);
    13. Book selectByPrimaryKey(Integer bid);
    14. List selectByIn(List bidIds);
    15. }

    然后我们Alt + Enter,自动补全实现方法:

    1. @Override
    2. public List selectByIn(List bidIds) {
    3. return bookMapper.selectByIn(bidIds);
    4. }

     再次Alt + Enter,建立一个测试方法进行测试:

    1. @Test
    2. public void selectByIn() {
    3. List bookIds = Arrays.asList(new Integer[]{31, 32, 33, 34});
    4. bookBiz.selectByIn(bookIds).forEach(System.out::println);
    5. }

    最后我们运行测试: 

    我们看到测试结果,查询到了,这就是关于forEach标签的使用,多去看看粘贴的代码。


    二、模糊查询

     2.1 mybatis模糊查询的三种形式

    和上一个使用forEach标签测试的方法实现步骤一样;

    先把三种方式的配置信息放到BookMapper.xml里面:

    1. <select id="selectBooksLike1" resultType="com.leaf.model.Book" parameterType="java.lang.String">
    2. select * from t_mvc_book where bname like #{bname}
    3. select>
    4. <select id="selectBooksLike2" resultType="com.leaf.model.Book" parameterType="java.lang.String">
    5. select * from t_mvc_book where bname like '${bname}'
    6. select>
    7. <select id="selectBooksLike3" resultType="com.leaf.model.Book" parameterType="java.lang.String">
    8. select * from t_mvc_book where bname like concat(concat('%',#{bname}),'%')
    9. select>

    然后还是一样的在自动生成的方法接口BookMapper.java里面也添加上三个方法:

    1. List selectBooksLike1(@Param("bname") String bname);
    2. List selectBooksLike2(@Param("bname") String bname);
    3. List selectBooksLike3(@Param("bname") String bname);

    然后再在我们自己写的实际使用的接口Bookbiz中也添加上:

    1. List selectBooksLike1(String bname);
    2. List selectBooksLike2(String bname);
    3. List selectBooksLike3(String bname);

    然后我们Alt + Enter,自动补全实现方法:

    1. @Override
    2. public List selectBooksLike1(String bname) {
    3. return bookMapper.selectBooksLike1(bname);
    4. }
    5. @Override
    6. public List selectBooksLike2(String bname) {
    7. return bookMapper.selectBooksLike2(bname);
    8. }
    9. @Override
    10. public List selectBooksLike3(String bname) {
    11. return bookMapper.selectBooksLike3(bname);
    12. }

      再次Alt + Enter,建立一个测试方法进行测试:

    1. @Test
    2. public void selectBooksLike1() {
    3. bookBiz.selectBooksLike1("%圣墟%").forEach(System.out::println);
    4. }

    我们先测试第一个: 结果已经出来了,我们看看他的日志信息,看看SQL语句具体是怎么实现的:

    我们记住这种写法,然后继续测试第二个:

    1. @Test
    2. public void selectBooksLike2() {
    3. bookBiz.selectBooksLike2("%圣墟%").forEach(System.out::println);
    4. }

     结果也查询出来了,但是我们看看SQL语句:

    我们可以发现,第一种和第二种的SQL并不一样!

    我们回过最开始的xml配置,仔细看看第一种和第二种的配置区别:

    这个时候我们如果把第二种方式的那个单引号去掉,就会报错:

    这个时候我们再次回想一下前面截图的两种方式使用的SQL语句方式,就可以明白,

    由于第二种SQL方式是直接用like '%关键字%'

    所以一旦配置文件那里没有单引号后,运行的时候就会缺失单引号导致报SQL的错误。

     所以我们得到一个结论

    在MyBatis里,实现模糊查询的两种符号 ${...} 和 #{...}区别

            ${...}:SQL中使用的是参数传递,会将传入的数据直接放到SQL中,一般用于传入数据库对象,例如传入表名;

            #{...}:SQL中使用的是占位符,会将传进来的数据都当做一个字符串,会对自动传入的数据加一个双引号,可以防止模糊查询的时候SQL注入。

            一般能用 # 的就别用 $

    然后我们再来测试一下第三种模糊查询的方式:

    第三种方式我们只需要传入纯参数就可以了,这种也是我们正常开发中写模糊查询的方式,不需要拼接%%就可以查询到。

    1. @Test
    2. public void selectBooksLike3() {
    3. bookBiz.selectBooksLike3("圣墟").forEach(System.out::println);
    4. }

    运行结果:


    三、mybatis对结果集的处理

    3.1 使用resultMap自定义返回值  

         3.1.1 使用resultMap返回自定义类型集合

         3.1.2 使用resultType返回List

         3.1.3 使用resultType返回单个对象

         3.1.4 使用resultType返回List,适用于多表查询返回结果集

         3.1.5 使用resultType返回Map,适用于多表查询返回单个结果集

           我们还是先把配置信息加到BookMapper.xml里面:

    1. <select id="list1" resultMap="BaseResultMap">
    2. select * from t_mvc_book
    3. select>
    4. <select id="list2" resultType="com.leaf.model.Book">
    5. select * from t_mvc_book
    6. select>
    7. <select id="list3" resultType="com.leaf.model.Book" parameterType="com.leaf.model.BookVo">
    8. select * from t_mvc_book where bid in
    9. <foreach collection="bookIds" open="(" close=")" separator="," item="bid">
    10. #{bid}
    11. foreach>
    12. select>
    13. <select id="list4" resultType="java.util.Map">
    14. select * from t_mvc_book
    15. select>
    16. <select id="list5" resultType="java.util.Map" parameterType="java.util.Map">
    17. select * from t_mvc_book where bid = #{bid}
    18. select>

       然后我们需要建立一个BookVo类:

    1. package com.leaf.model;
    2. import java.util.List;
    3. /**
    4. * @author Leaf
    5. * @site 2977819715
    6. * @company 玉渊工作室
    7. * @create  2022-08-12 11:27
    8. */
    9. public class BookVo extends Book {
    10. private List bookIds;
    11. public List getBookIds() {
    12. return bookIds;
    13. }
    14. public void setBookIds(List bookIds) {
    15. this.bookIds = bookIds;
    16. }
    17. }

    如果是单表的情况下,resultType与resultMap都可以使用;

    不管返回一条数据,还是多条数据,都应该用java.util.Map进行接收;

    如果是多条数据,那么返回值List

    在自动生成的Bookmapper.java里面添加对应测试的方法:

    1. //list1 list2的结论:对于单表查询而言,可以用它resultType/resultMap接收;
    2. //但是多表必须用resultMap接收
    3. List list1();
    4. List list2();
    5. //如果要传入多个查询参数就必须以对象的方式进行传递
    6. List list3(BookVo vo);
    7. //如果是返回一条数据,那么返回值Map
    8. //如果是多条数据,那么返回值List
    9. List list4();
    10. Map list5(Map map);

     同时还要添加到自己写的BookBiz里面:

    1. List list1();
    2. List list2();
    3. List list3(BookVo vo);
    4. List list4();
    5. Map list5(Map map);

    然后Alt + Enter,补全实现方法:

    1. @Override
    2. public List list1() {
    3. return bookMapper.list1();
    4. }
    5. @Override
    6. public List list2() {
    7. return bookMapper.list2();
    8. }
    9. @Override
    10. public List list3(BookVo vo) {
    11. return bookMapper.list3(vo);
    12. }
    13. @Override
    14. public List list4() {
    15. return bookMapper.list4();
    16. }
    17. @Override
    18. public Map list5(Map map) {
    19. return bookMapper.list5(map);
    20. }

     然后我们再次Alt + Enter,建立测试方法:BookBizimplTest

    1. @Test
    2. public void list1() {
    3. bookBiz.list1().forEach(System.out::println);
    4. }
    5. @Test
    6. public void list2() {
    7. bookBiz.list2().forEach(System.out::println);
    8. }
    9. @Test
    10. public void list3() {
    11. BookVo vo=new BookVo();
    12. vo.setBookIds(Arrays.asList(new Integer[]{31,32,33,34}));
    13. bookBiz.list3(vo).forEach(System.out::println);
    14. }
    15. @Test
    16. public void list4() {
    17. bookBiz.list4().forEach(System.out::println);
    18. }
    19. @Test
    20. public void list5() {
    21. Map map=new HashMap();
    22. map.put("bid",32);
    23. System.out.println(bookBiz.list5(map));
    24. }

    测试结果: 

    List1、List2:

    对于单表查询而言,可以用它resultType/resultMap接收,但是多表必须用resultMap接收

     

    List3:

    要传入多个查询参数,就必须以对象的方式进行传递。

     List4: 

    如果返回的是一条数据,那么返回值就为Map;
    如果返回的是多条数据,那么返回值就为List
     

    List5:

    四、分页查询

            4.1 第三方分页插件

            由于MyBatis的分页功能很弱,它是基于内存的分页;

    注:

    就是查出所有记录,再按偏移量offset和边界limit取结果,

    在大数据量的情况下这样的分页基本上是没有用的。

            所以我们需要重写MyBatis的分页

            我们找到一个第三方分页的插件,以此作为基础进行使用;

            插件使用步骤:

    1、导入pom依赖


        com.github.pagehelper
        pagehelper
        5.1.2

    2、Mybatis.cfg.xml配置拦截器 


       
       
       

    3、使用PageHelper进行分页 

    4、处理分页结果

             我们完善前面写的BookVo

    1. package com.leaf.model;
    2. import java.util.List;
    3. /**
    4. * @author Leaf
    5. * @site 2977819715
    6. * @company 玉渊工作室
    7. * @create  2022-08-12 11:27
    8. */
    9. public class BookVo extends Book {
    10. private List bookIds;
    11. private int min;
    12. private int max;
    13. public int getMax() {
    14. return max;
    15. }
    16. public void setMax(int max) {
    17. this.max = max;
    18. }
    19. public int getMin() {
    20. return min;
    21. }
    22. public void setMin(int min) {
    23. this.min = min;
    24. }
    25. public List getBookIds() {
    26. return bookIds;
    27. }
    28. public void setBookIds(List bookIds) {
    29. this.bookIds = bookIds;
    30. }
    31. }

            建立分页工具类:PageBean

    1. package com.leaf.pagination.entity;
    2. import javax.servlet.http.HttpServletRequest;
    3. import java.io.Serializable;
    4. import java.util.Map;
    5. /**
    6. * @author Leaf
    7. * @site 2977819715
    8. * @company 玉渊工作室
    9. * @create  2022-08-12 13:11
    10. */
    11. public class PageBean implements Serializable {
    12. private static final long serialVersionUID = 2422581023658455731L;
    13. //页码
    14. private int page=1;
    15. //每页显示记录数
    16. private int rows=10;
    17. //总记录数
    18. private int total=0;
    19. //是否分页
    20. private boolean isPagination=true;
    21. //上一次的请求路径
    22. private String url;
    23. //获取所有的请求参数
    24. private Map map;
    25. public PageBean() {
    26. super();
    27. }
    28. //设置请求参数
    29. public void setRequest(HttpServletRequest req) {
    30. String page=req.getParameter("page");
    31. String rows=req.getParameter("rows");
    32. String pagination=req.getParameter("pagination");
    33. this.setPage(page);
    34. this.setRows(rows);
    35. this.setPagination(pagination);
    36. this.url=req.getContextPath()+req.getServletPath();
    37. this.map=req.getParameterMap();
    38. }
    39. public String getUrl() {
    40. return url;
    41. }
    42. public void setUrl(String url) {
    43. this.url = url;
    44. }
    45. public Map getMap() {
    46. return map;
    47. }
    48. public void setMap(Map map) {
    49. this.map = map;
    50. }
    51. public int getPage() {
    52. return page;
    53. }
    54. public void setPage(int page) {
    55. this.page = page;
    56. }
    57. public void setPage(String page) {
    58. if(null!=page&&!"".equals(page.trim()))
    59. this.page = Integer.parseInt(page);
    60. }
    61. public int getRows() {
    62. return rows;
    63. }
    64. public void setRows(int rows) {
    65. this.rows = rows;
    66. }
    67. public void setRows(String rows) {
    68. if(null!=rows&&!"".equals(rows.trim()))
    69. this.rows = Integer.parseInt(rows);
    70. }
    71. public int getTotal() {
    72. return total;
    73. }
    74. public void setTotal(int total) {
    75. this.total = total;
    76. }
    77. public void setTotal(String total) {
    78. this.total = Integer.parseInt(total);
    79. }
    80. public boolean isPagination() {
    81. return isPagination;
    82. }
    83. public void setPagination(boolean isPagination) {
    84. this.isPagination = isPagination;
    85. }
    86. public void setPagination(String isPagination) {
    87. if(null!=isPagination&&!"".equals(isPagination.trim()))
    88. this.isPagination = Boolean.parseBoolean(isPagination);
    89. }
    90. /**
    91. * 获取分页起始标记位置
    92. * @return
    93. */
    94. public int getStartIndex() {
    95. //(当前页码-1)*显示记录数
    96. return (this.getPage()-1)*this.rows;
    97. }
    98. /**
    99. * 末页
    100. * @return
    101. */
    102. public int getMaxPage() {
    103. int totalpage=this.total/this.rows;
    104. if(this.total%this.rows!=0)
    105. totalpage++;
    106. return totalpage;
    107. }
    108. /**
    109. * 下一页
    110. * @return
    111. */
    112. public int getNextPage() {
    113. int nextPage=this.page+1;
    114. if(this.page>=this.getMaxPage())
    115. nextPage=this.getMaxPage();
    116. return nextPage;
    117. }
    118. /**
    119. * 上一页
    120. * @return
    121. */
    122. public int getPreivousPage() {
    123. int previousPage=this.page-1;
    124. if(previousPage<1)
    125. previousPage=1;
    126. return previousPage;
    127. }
    128. @Override
    129. public String toString() {
    130. return "PageBean [page=" + page + ", rows=" + rows + ", total=" + total + ", isPagination=" + isPagination
    131. + "]";
    132. }
    133. }

    然后在BookMapper里面添加分页的接口方法:

    1. //分页
    2. List listPager(Map map);

     在我们自己写的接口BookBiz里面也同样添加:

     List listPager(Map map, PageBean pageBean);

    然后Alt + Enter,补全实现类的实现方法: 

    1. @Override
    2. public List listPager(Map map, PageBean pageBean) {
    3. //pageHelper分页插件相关代码
    4. if(pageBean != null && pageBean.isPagination()){
    5. PageHelper.startPage(pageBean.getPage(),pageBean.getRows());
    6. }
    7. List maps=bookMapper.listPager(map);
    8. if(pageBean != null && pageBean.isPagination()){
    9. //处理查询结果的前提,是需要分页的
    10. PageInfo info=new PageInfo(maps);
    11. pageBean.setTotal(info.getTotal()+"");
    12. }
    13. return maps;
    14. }

    然后就可以建立测试方法开始测试啦:

    1. @Test
    2. public void listPager(){
    3. Map map=new HashMap();
    4. map.put("bname","圣墟");
    5. bookBiz.listPager(map,pageBean).forEach(System.out::println);
    6. //查询出第二页的20条数据
    7. PageBean pageBean=new PageBean();
    8. pageBean.setPage(2);
    9. pageBean.setRows(20);
    10. bookBiz.listPager(map,pageBean).forEach(System.out::println);
    11. }

            4.2 集成mybatis进行使用


    五、对特殊字符的处理

    5.1 配置BookMapper.xml

    我们配置下面这个:

    1. <select id="list6" resultType="com.leaf.model.Book" parameterType="com.leaf.model.BookVo">
    2. select * from t_mvc_book
    3. <where>
    4. <if test="null != min and min != ''">
    5. if>
    6. <if test="null != max and max != ''">
    7. price ]]>
    8. if>
    9. where>
    10. select>
    11. <select id="list7" resultType="com.leaf.model.Book" parameterType="com.leaf.model.BookVo">
    12. select * from t_mvc_book
    13. <where>
    14. <if test="null != min and min != ''">
    15. and #{min} < price
    16. if>
    17. <if test="null != max and max != ''">
    18. and #{max} > price
    19. if>
    20. where>
    21. select>

    然后就都是老步骤了,两个接口类方法:

    1. List list6(BookVo vo);
    2. List list7(BookVo vo);

    实现类:

    1. @Override
    2. public List list6(BookVo vo) {
    3. return bookMapper.list6(vo);
    4. }
    5. @Override
    6. public List list7(BookVo vo) {
    7. return bookMapper.list7(vo);
    8. }

    测试方法:

    1. @Test
    2. public void list6() {
    3. BookVo vo=new BookVo();
    4. vo.setMax(45);
    5. vo.setMin(35);
    6. bookBiz.list6(vo).forEach(System.out::println);
    7. }
    8. @Test
    9. public void list7() {
    10. BookVo vo=new BookVo();
    11. vo.setMax(45);
    12. vo.setMin(35);
    13. bookBiz.list7(vo).forEach(System.out::println);
    14. }

    OK

  • 相关阅读:
    msvcp120.dll缺失的解决方法与作用介绍
    微信小程序--》小程序—自定义组件使用
    k8s-mysql主从部署
    第三方模块远程注入到软件中引发软件异常的若干实战案例分享
    暗云III木马技术分析
    底层概念的重要意义
    对自动化测试的一些展望与理解
    大数据运维实战第十七课 日志收集、分析过滤工具 Logstash应用实战
    【BOOST C+容器专题03】【10】Boost.Container
    Java之泛型系列--构造方法使用泛型(有示例)
  • 原文地址:https://blog.csdn.net/qq_63492318/article/details/126296747