站在巨人的肩膀上

https://www.cnblogs.com/esileme/p/7565184.html

环境:Spring 4.2.1 Mybatis 3.2.8 pagehelper 5.1.2

Mybatis官方教程:https://github.com/pagehelper/Mybatis-PageHelper/blob/d5947437cc6272cb3f1b186cadee74f1f8072cbb/wikis/zh/HowToUse.md

  • 加入pageHelper依赖

     <dependency>
    <groupId>com.github.pagehelper</groupId>
    <artifactId>pagehelper</artifactId>
    <version>5.1.2</version>
    </dependency>
  • 在Mybatis-config.xml中配置pageHelper的拦截器

     <plugins>
    <!-- com.github.pagehelper为PageHelper类所在包名 -->
    <plugin interceptor="com.github.pagehelper.PageInterceptor">
    <!-- 使用下面的方式配置参数,后面会有所有的参数介绍 -->
    <property name="helperDialect" value="mysql" />
    <property name="reasonable" value="true" />
    <property name="supportMethodsArguments" value="true" />
    <property name="autoRuntimeDialect" value="true" /> </plugin> </plugins>
  • 调用startPage()分页方法,在获取数据后将数据放到PageInfo对象中。

     PageHelper.startPage(page, pageSize);
    List<Map<String, Object>> list = itemMapper.getItems();
    PageInfo<Map<String, Object>> pageInfor = new PageInfo<Map<String, Object>>(list);

即可对数据进行分页处理。

Mybatis使用pageHelper分页出现的问题

在pageHelper的文档中,我们可以看到这样的提示分页插件不支持嵌套结果映射https://github.com/pagehelper/Mybatis-PageHelper/blob/d5947437cc6272cb3f1b186cadee74f1f8072cbb/wikis/zh/Important.md),由于嵌套结果方式会导致结果集被折叠,因此分页查询的结果在折叠后总数会减少,所以无法保证分页结果数量正确。这时候对于一些嵌套查询来说使用通用pageHelper已经不能用了,因此需要我们通过手动分别查询分页数据以及分页数据的映射数据。

通过使用一个例子对pageHelper进行测试不兼容嵌套查询的结果映射:

 <resultMap type="java.util.Map" id="fatherMap">
<result column="orderId" property="orderId" />
<result column="payment" property="payment" />
<result column="buyer_nick" property="buyerNick" />
<result column="shipping_code" property="shippingCode" />
<result column="state" property="state" />
</resultMap> <resultMap type="java.util.Map" id="orderMap" extends="fatherMap"> <collection property="items" javaType="ArrayList" resultMap="itemMap" /> </resultMap> <select id="getOrdersByUserId" resultMap="orderMap"> SELECT o.order_id AS orderId,o.payment ,o.buyer_nick,o.shipping_code
,o.status AS orderState,i.title ,i.price FROM tb_order AS o,tb_item AS
i,tb_order_item AS oi WHERE o.user_id=#{userId} AND
oi.order_id=o.order_id AND i.id=oi.item_id </select>
   /*Service层代码*/
public List<Map<String, Object>> getOrdersByUserId(String userId) { PageHelper.startPage(1, 2);
List<Map<String, Object>> list = orderMapper.getOrdersByUserId(userId);
return list; }

通过嵌套查询返回一个Map对象并嵌套了另一个表中的多条数据(List集合),在取出数据后,可以看到这样的结果:

可以发现,pageHelper对我们的子数据进行了分页,我们要两条数据且只是出现了一条数据,这个时候,就只有通过分别查询来取出数据了。

首先,对订单数据不进行关联查询,先把数据取出来,然后根据取出来的订单对象,再把关联的信息查出来,这样就可以使用pageHelper来进行操作了。在Service层的代码如下:

    public List<Map<String, Object>> getAnotherOrdersByUserId(String userId) {
PageHelper.startPage(1, 2);
List<Map<String, Object>> orderMaps = orderMapper.getAnotherOrdersByUserId(userId);
for (Map<String, Object> map : orderMaps) {
String orderId = (String) map.get("orderId");
List<Map<String, Object>> items = orderMapper.getItems(orderId);
map.put("items", items);
}
PageInfo pageInfor = new PageInfo(orderMaps);
System.out.println(pageInfor.getPageNum());
System.out.println(pageInfor.getTotal()); return orderMaps;
}

mapper的代码:

<select id="getAnotherOrdersByUserId" resultMap="fatherMap">

    SELECT o.order_id AS orderId,o.payment ,o.buyer_nick,o.shipping_code
,o.status AS orderState FROM tb_order AS o WHERE o.user_id=#{userId} </select> <select id="getItems" resultMap="itemMap"> SELECT item.title ,item.price FROM tb_order_item AS oi ,tb_item AS item
WHERE oi.order_id=#{orderId} AND oi.item_id=item.id </select>

这样通过分别查询就可以避免了pageHelper的联合查询的问题。

正确的数据显示:

Demo:https://github.com/esileme/MybatisPager

PageHelper分页原理

在Mybatis中,提供了一个Interceptor接口,Mybatis的PageInterceptor类实现了Interceptor接口,在这个接口中,MyBatis允许我们拦截四个方法(ParameterHandler、ResultSetHandler、StatementHandler、Executor),可以对这四个方法进行拦截并实现相应的操作。

我们可以自定义实现Interceptor方式对Mybatis返回为Map的对象进行数据库数据与pojo对象映射的操作:

  • 在Mybatis-config.xml中配置

    <plugin interceptor="com.yl.page.intercpter.CameHumpIntercpter"></plugin>
    code
  • package com.yl.page.intercpter;
    
    import java.sql.Statement;
    import java.util.HashSet;
    import java.util.List;
    import java.util.Map;
    import java.util.Properties;
    import java.util.Set; import org.apache.ibatis.executor.resultset.ResultSetHandler;
    import org.apache.ibatis.plugin.Interceptor;
    import org.apache.ibatis.plugin.Intercepts;
    import org.apache.ibatis.plugin.Invocation;
    import org.apache.ibatis.plugin.Plugin;
    import org.apache.ibatis.plugin.Signature; @Intercepts( @Signature(type = ResultSetHandler.class, method = "handleResultSets", args = { Statement.class }))
    public class CameHumpIntercpter implements Interceptor { public Object intercept(Invocation invocation) throws Throwable {
    List<Object> values = (List<Object>) invocation.proceed();
    for (Object object : values) { if (object instanceof Map) {
    processMap((Map) object); } else {
    break;
    }
    }
    return values;
    } public Object plugin(Object target) { return Plugin.wrap(target, this);
    } // 获取配置参数
    public void setProperties(Properties properties) { } private void processMap(Map<String, Object> map) {
    Set<String> keySet = new HashSet<String>(map.keySet());
    for (String key : keySet) { if ((key.charAt(0) >= 'A' && key.charAt(0) <= 'Z') || key.indexOf("_") >= 0) { Object value = map.get(key);
    map.remove(key);
    map.put(undelineToNomall(key), value);
    }
    } } public String undelineToNomall(String inputString) {
    StringBuilder buder = new StringBuilder();
    boolean nextUpperCase = false;
    for (int i = 0; i < inputString.length(); i++) { char c = inputString.charAt(i);
    if (c == '_') {
    if (buder.length() > 0) {
    nextUpperCase = true;
    } } else { if (nextUpperCase) {
    buder.append(Character.toUpperCase(c)); nextUpperCase = false;
    } else {
    buder.append(Character.toLowerCase(c));
    } } } return buder.toString();
    } }

Mybatis中分页存在的坑1的更多相关文章

  1. mybatis中集成sharing-jdbc采坑

    1. mybatis中集成sharing-jdbc采坑 1.1. 错误信息 Caused by: org.apache.ibatis.binding.BindingException: Invalid ...

  2. mybatis中分页查询

    1 如果在查询方法中有多个参数,可以使用map对象将所有数据都存储进去.比如分页查询,需要用到两个参数,可以将这两个参数包装到map中. 例子:分页查询 dao层方法 public List<S ...

  3. mybatis中分页插件PageHelper的使用

    转载博客:http://blog.csdn.net/u012728960/article/details/50791343

  4. MyBatis中log4j 和 参数 和 分页和别名 功能

    1.配置全局文件,注意各个配置标签的顺序 properties?, settings?, typeAliases?, typeHandlers?, objectFactory?,   objectWr ...

  5. Oracle使用MyBatis中RowBounds实现分页查询

    Oracle中分页查询因为存在伪列rownum,sql语句写起来较为复杂,现在介绍一种通过使用MyBatis中的RowBounds进行分页查询,非常方便. 使用MyBatis中的RowBounds进行 ...

  6. Mybatis中使用PageHelper插件进行分页

    分页的场景比较常见,下面主要介绍一下使用PageHelper插件进行分页操作: 一.概述: PageHelper支持对mybatis进行分页操作,项目在github地址: https://github ...

  7. mybatis中的分页插件

    1.Mybatis的分页plugin实现原理 2.具体步骤 第一步.导入到pom.xml文件中依赖包 第二步.配置插件(必需) 在mybatisConfig.xml文件中配置以下代码 代码位置:在en ...

  8. mybatis中Oracle分页语句的写法

    最近一段时间使用oracle数据库查询分页, 用的是springboot. Oracle数据库中没有像mysql中limit的写法, 只能换其他方式写. 考虑到oracle中的ROWNUM变量, 使用 ...

  9. 记自己在mybatis中设置jdbcType的一个坑

    项目是用ssm搭建的.主要是为app数据接口.其中有一个需求就app想要查询一段时间内某个用户的测量信息,所以app给我后端传递了3个参数,分别是appuserId(String),startDate ...

随机推荐

  1. JS中正则匹配开头不带空格,结尾也不带空格的字符串

    在做项目的时候,要求限制SSID的长度.以及开头和结尾不能是空格. var reg = /^\S.{0,30}\S$/ "$$$  $$".match(reg);   ==> ...

  2. Git简介和安装

    一.什么是Git? Git是一个分布式版本控制系统,客户端并不只提取最新版本的文件快照,而是把代码仓库完整地镜像下来.如图所示: 任何一处的服务器或者个人机发生故障,都可以用其它机器的任何一个镜像出来 ...

  3. [Java SE] 字符串连接

    Java 支持多种字符串连接方式,总结如下: package cn.spads.tool.string; import java.text.MessageFormat; /** * <b> ...

  4. some base knowledge

    har类型的长度被定义为一个8位字节,这很简单. short类型的长度至少为两字节.在有些计算机上,对于有些编译程序,short类型的长度可能为4字节,或者更长. int类型是一个整数的“自然”大小, ...

  5. curl上传图片(同域上传)

    先研究完curl同域上传图片然后再研究curl跨域上传,先把同域上传代码总结如下: index.php <?php //图片上传 $url = "http://192.168.1.23 ...

  6. Linux 查看服务状态(服务与进程)

    原文地址:http://blog.csdn.net/weixin_37979944/article/details/79029188 指令(instruction): 计算机实际上可以做的事情实质上非 ...

  7. 机器学习: Canonical Correlation Analysis 典型相关分析

    Canonical Correlation Analysis(CCA)典型相关分析也是一种常用的降维算法.我们知道,PCA(Principal Component Analysis) 主分量分析将数据 ...

  8. Oracle数据常用操作

    将用逗号隔开字段拆分成两行: select * from mp_fs_file_info a,dm_process_upload b where instr(b.attachment,a.file_i ...

  9. Turn your Session into FlushMode.COMMIT/AUTO or remove 'readOnly' marker from transaction definition.

  10. poj3417 Network——LCA+树上差分

    题目:http://poj.org/problem?id=3417 根据一条边被几个环覆盖来判断能不能删.有几种情况等: 用树上差分,终点 s++,LCA s-=2,统计时计算子树s值的和即可: 用S ...