Mybatis分页和Spring的集成
Mybatis分页和Spring的集成
miemiedev 发表于4年前
Mybatis分页和Spring的集成
  • 发表于 4年前
  • 阅读 45852
  • 收藏 109
  • 点赞 22
  • 评论 108

腾讯云 十分钟定制你的第一个小程序>>>   

写了一个Mybatis分页控件,在这记录一下使用方式。

在Maven中加入依赖:

<dependencies>
  ...
    <dependency>
        <groupId>com.github.miemiedev</groupId>
        <artifactId>mybatis-paginator</artifactId>
        <version>1.2.17</version>
    </dependency>
 ...
</dependencies>


Mybatis配置文件添加分页插件:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE configuration
        PUBLIC "-//ibatis.apache.org//DTD Config 3.0//EN"
        "http://ibatis.apache.org/dtd/ibatis-3-config.dtd">
<configuration>
    <plugins>
        <plugin interceptor="com.github.miemiedev.mybatis.paginator.OffsetLimitInterceptor">
            <property name="dialectClass" value="com.github.miemiedev.mybatis.paginator.dialect.OracleDialect"/>
        </plugin>
    </plugins>
</configuration>


创建一个查询,内容可以是任何Mybatis表达式,包括foreach和if等:

<select id="findByCity" resultType="map">
    select * from TEST_USER where city = #{city};
</select>


Dao中的方法或许是这样(用接口也是类似):

public List findByCity(String city, PageBounds pageBounds){

    Map<String, Object> params = new HashMap<String, Object>();
    params.put("city",city);

    return getSqlSession().selectList("db.table.user.findByCity", params, pageBounds);
}


调用方式(分页加多列排序):

int page = 1; //页号
int pageSize = 20; //每页数据条数
String sortString = "age.asc,gender.desc";//如果你想排序的话逗号分隔可以排序多列
PageBounds pageBounds = new PageBounds(page, pageSize , Order.formString(sortString));
List list = findByCity("BeiJing",pageBounds);

//获得结果集条总数
PageList pageList = (PageList)list;
System.out.println("totalCount: " + pageList.getPaginator().getTotalCount());

PageList类是继承于ArrayList的,这样Dao中就不用为了专门分页再多写一个方法。

使用PageBounds这个对象来控制结果的输出,常用的使用方式一般都可以通过构造函数来配置。

new PageBounds();//默认构造函数不提供分页,返回ArrayList
new PageBounds(int limit);//取TOPN操作,返回ArrayList
new PageBounds(Order... order);//只排序不分页,返回ArrayList

new PageBounds(int page, int limit);//默认分页,返回PageList
new PageBounds(int page, int limit, Order... order);//分页加排序,返回PageList
new PageBounds(int page, int limit, List<Order> orders, boolean containsTotalCount);//使用containsTotalCount来决定查不查询totalCount,即返回ArrayList还是PageList


========================================= 

如果用的是Spring MVC的话可以把JSON的配置写成这样:

<mvc:annotation-driven>
    <mvc:message-converters register-defaults="true">
        <bean class="org.springframework.http.converter.StringHttpMessageConverter"> 
            <constructor-arg value="UTF-8" />        
        </bean>
        <bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter">
            <property name="objectMapper">
                <bean class="com.github.miemiedev.mybatis.paginator.jackson2.PageListJsonMapper" />
            </property>
        </bean>
    </mvc:message-converters>
</mvc:annotation-driven>


那么在Controller就可以这样用了:

@ResponseBody
@RequestMapping(value = "/findByCity.json")
public List findByCity(@RequestParam String city,
                 @RequestParam(required = false,defaultValue = "1") int page,
                 @RequestParam(required = false,defaultValue = "30") int limit,
                 @RequestParam(required = false) String sort,
                 @RequestParam(required = false) String dir) {
    return userService.findByCity(city, new PageBounds(page, limit, Order.create(sort,dir)));
}


然后序列化后的JSON字符串就会变成这样的:

{
    "items":[
        {"NAME":"xiaoma","AGE":30,"GENDER":1,"ID":3,"CITY":"BeiJing"},
        {"NAME":"xiaoli","AGE":30,"SCORE":85,"GENDER":1,"ID":1,"CITY":"BeiJing"},
        {"NAME":"xiaowang","AGE":30,"SCORE":92,"GENDER":0,"ID":2,"CITY":"BeiJing"},
        {"NAME":"xiaoshao","AGE":30,"SCORE":99,"GENDER":0,"ID":4,"CITY":"BeiJing"}
    ],
    "slider": [1, 2, 3, 4, 5, 6, 7],
    "hasPrePage": false,
    "startRow": 1,
    "offset": 0,
    "lastPage": false,
    "prePage": 1,
    "hasNextPage": true,
    "nextPage": 2,
    "endRow": 30,
    "totalCount": 40351,
    "firstPage": true,
    "totalPages": 1346,
    "limit": 30,
    "page": 1
}


=========================================

在SpringMVC中使用JSTL的话可以参考一下步骤(懒人用法)

在Spring配置文件中加入拦截器,或则参考拦截器实现定义自己的拦截器

<mvc:interceptors>
    <mvc:interceptor>
        <mvc:mapping path="/**" />
        <bean class="com.github.miemiedev.mybatis.paginator.springmvc.PageListAttrHandlerInterceptor" />
    </mvc:interceptor>
</mvc:interceptors>

然后Controller方法可以这样写

@RequestMapping(value = "/userView.action")
public ModelAndView userView(@RequestParam String city,
                 @RequestParam(required = false,defaultValue = "1") int page,
                 @RequestParam(required = false,defaultValue = "30") int limit,
                 @RequestParam(required = false) String sort,
                 @RequestParam(required = false) String dir) {
    List users = userService.findByCity(city, new PageBounds(page, limit, Order.create(sort,dir)));
    return new ModelAndView("account/user","users", users);
}

JSP中就可以这样用了,拦截器会将PageList分拆添加Paginator属性,默认命名规则为"原属性名称"+"Paginator"

<table>
    <c:forEach items="${users}" var="user">
        <tr>
            <td>${user['ID']}</td>
            <td>${user['NAME']}</td>
            <td>${user['AGE']}</td>
        </tr>
    </c:forEach>
</table>
上一页: ${usersPaginator.prePage} 
当前页: ${usersPaginator.page} 
下一页: ${usersPaginator.nextPage} 
总页数: ${usersPaginator.totalPages} 
总条数: ${usersPaginator.totalCount} 
更多属性参考Paginator类提供的方法

=========================================

如果用如下方法设置pageBounds,当前这个查询就可以用两个线程同时查询list和totalCount

pageBounds.setAsyncTotalCount(true);

如果所有的分页查询都是用异步的方式查询list和totalCount,可以在插件配置加入asyncTotalCount属性

<plugin interceptor="com.github.miemiedev.mybatis.paginator.OffsetLimitInterceptor">
    <property name="dialectClass" value="com.github.miemiedev.mybatis.paginator.dialect.OracleDialect"/>
    <property name="asyncTotalCount" value="true"/>
</plugin>

但是你仍然可以用下面代码强制让这个查询不用异步

pageBounds.setAsyncTotalCount(false);

当然需要注意的是,只要你用到了异步查询,由于里面使用了线程池,所以在使用时就要加入清理监听器,以便在停止服务时关闭线程池。需要在web.xml中加入

<listener>
    <listener-class>com.github.miemiedev.mybatis.paginator.CleanupMybatisPaginatorListener</listener-class>
</listener>


完。

共有 人打赏支持
miemiedev
粉丝 70
博文 2
码字总数 1543
作品 2
评论 (108)
飞天遇见猪
这个只是针对oracle吗?
miemiedev

引用来自“飞天遇见猪”的评论

这个只是针对oracle吗?

mysql和db2也行,看一下dialect包里的类
飞天遇见猪
OK,有空研究下源码~~
学习奋斗
插件不错,请问带查询的分页怎么设置?
无悔这一生
您好,我在PageList pageList = (PageList)list; 强制转换这里报错了,我有点不解,为啥这里来个强转,就能取到Paginator对象? 我debug了下,list就是一个普通的List对象,如何能变出一个Paginator对象?
无悔这一生
好了 我自己的问题!
miemiedev

引用来自“唐文韬”的评论

您好,我在PageList pageList = (PageList)list; 强制转换这里报错了,我有点不解,为啥这里来个强转,就能取到Paginator对象? 我debug了下,list就是一个普通的List对象,如何能变出一个Paginator对象?

区别在于containsTotalCount参数是否为true。有时候只想取TopN就可以不查询记录总数了。
无悔这一生

引用来自“miemiedev”的评论

引用来自“唐文韬”的评论

您好,我在PageList pageList = (PageList)list; 强制转换这里报错了,我有点不解,为啥这里来个强转,就能取到Paginator对象? 我debug了下,list就是一个普通的List对象,如何能变出一个Paginator对象?

区别在于containsTotalCount参数是否为true。有时候只想取TopN就可以不查询记录总数了。

谢谢您的答复!
javaee_jsp
Method [list] returned [[{id=1, email=javaee_jsp@163.com, name=HOBO, telphone=13774360854}, {id=3, email=111, name=11, telphone=111}, {id=4, email=44, name=33, telphone=33}, {id=5, email=ee, name=ee, telphone=ee}, {id=6, email=22, name=33, telphone=11}, {id=7, email=22, name=33, telphone=11}, {id=8, email=22, name=33, telphone=11}, {id=9, email=22, name=33, telphone=11}]] 你好,我运行出来json数据···
miemiedev

引用来自“javaee_jsp”的评论

Method [list] returned [[{id=1, email=javaee_jsp@163.com, name=HOBO, telphone=13774360854}, {id=3, email=111, name=11, telphone=111}, {id=4, email=44, name=33, telphone=33}, {id=5, email=ee, name=ee, telphone=ee}, {id=6, email=22, name=33, telphone=11}, {id=7, email=22, name=33, telphone=11}, {id=8, email=22, name=33, telphone=11}, {id=9, email=22, name=33, telphone=11}]] 你好,我运行出来json数据···

你是想说什么?
javaee_jsp

引用来自“miemiedev”的评论

引用来自“javaee_jsp”的评论

Method [list] returned [[{id=1, email=javaee_jsp@163.com, name=HOBO, telphone=13774360854}, {id=3, email=111, name=11, telphone=111}, {id=4, email=44, name=33, telphone=33}, {id=5, email=ee, name=ee, telphone=ee}, {id=6, email=22, name=33, telphone=11}, {id=7, email=22, name=33, telphone=11}, {id=8, email=22, name=33, telphone=11}, {id=9, email=22, name=33, telphone=11}]] 你好,我运行出来json数据···

你是想说什么?

我想说json数据,输出不正确。、
miemiedev

引用来自“javaee_jsp”的评论

引用来自“miemiedev”的评论

引用来自“javaee_jsp”的评论

Method [list] returned [[{id=1, email=javaee_jsp@163.com, name=HOBO, telphone=13774360854}, {id=3, email=111, name=11, telphone=111}, {id=4, email=44, name=33, telphone=33}, {id=5, email=ee, name=ee, telphone=ee}, {id=6, email=22, name=33, telphone=11}, {id=7, email=22, name=33, telphone=11}, {id=8, email=22, name=33, telphone=11}, {id=9, email=22, name=33, telphone=11}]] 你好,我运行出来json数据···

你是想说什么?

我想说json数据,输出不正确。、

你看一下你输出的对象是不是叫PageList,如果是的话你看一下你的spring mvc的配置有没有加上面的转换器。如果你用的不是spring mvc需要自己手动转换数据。
javaee_jsp

引用来自“miemiedev”的评论

引用来自“javaee_jsp”的评论

引用来自“miemiedev”的评论

引用来自“javaee_jsp”的评论

Method [list] returned [[{id=1, email=javaee_jsp@163.com, name=HOBO, telphone=13774360854}, {id=3, email=111, name=11, telphone=111}, {id=4, email=44, name=33, telphone=33}, {id=5, email=ee, name=ee, telphone=ee}, {id=6, email=22, name=33, telphone=11}, {id=7, email=22, name=33, telphone=11}, {id=8, email=22, name=33, telphone=11}, {id=9, email=22, name=33, telphone=11}]] 你好,我运行出来json数据···

你是想说什么?

我想说json数据,输出不正确。、

你看一下你输出的对象是不是叫PageList,如果是的话你看一下你的spring mvc的配置有没有加上面的转换器。如果你用的不是spring mvc需要自己手动转换数据。

我看到你写的例子了,你里面用到mmGrid ,我也跟着试了下是可以的。
但是我用jQuery easy UI json 数据不对了。
miemiedev

引用来自“javaee_jsp”的评论

引用来自“miemiedev”的评论

引用来自“javaee_jsp”的评论

引用来自“miemiedev”的评论

引用来自“javaee_jsp”的评论

Method [list] returned [[{id=1, email=javaee_jsp@163.com, name=HOBO, telphone=13774360854}, {id=3, email=111, name=11, telphone=111}, {id=4, email=44, name=33, telphone=33}, {id=5, email=ee, name=ee, telphone=ee}, {id=6, email=22, name=33, telphone=11}, {id=7, email=22, name=33, telphone=11}, {id=8, email=22, name=33, telphone=11}, {id=9, email=22, name=33, telphone=11}]] 你好,我运行出来json数据···

你是想说什么?

我想说json数据,输出不正确。、

你看一下你输出的对象是不是叫PageList,如果是的话你看一下你的spring mvc的配置有没有加上面的转换器。如果你用的不是spring mvc需要自己手动转换数据。

我看到你写的例子了,你里面用到mmGrid ,我也跟着试了下是可以的。
但是我用jQuery easy UI json 数据不对了。

我上面说的那两点你检查过没?
ponta
  @ResponseBody
  @RequestMapping(value = "/getAllUsersByPage.json",method=RequestMethod.GET)
  public List<Users> getAllUsersByPage(
   @RequestParam(required = false,defaultValue = "1") int page,
   @RequestParam(required = false,defaultValue = "30") int limit,
   @RequestParam(required = false) String sort,
   @RequestParam(required = false) String dir) {
    PageBounds pageBounds = new PageBounds(page, limit, Order.create(sort, dir));
    pageBounds.setAsyncTotalCount(true);
   return loginService.getAllUsersByPage(pageBounds);
  }
你好,请问我的返回结果为什么没有 totalCount 呢?

[{"userid":7,"username":"user6","password":"pass2","enabled":1,"issys":1,"user_desc":"1"},{"userid":6,"username":"user5","password":"pass2","enabled":1,"issys":1,"user_desc":"1"},{"userid":5,"username":"user4","password":"pass2","enabled":1,"issys":1,"user_desc":"1"}]
ponta
pageBounds.setContainsTotalCount(true); 这个设置了也不行啊。求解!
miemiedev

引用来自“ponta”的评论

  @ResponseBody
  @RequestMapping(value = "/getAllUsersByPage.json",method=RequestMethod.GET)
  public List<Users> getAllUsersByPage(
   @RequestParam(required = false,defaultValue = "1") int page,
   @RequestParam(required = false,defaultValue = "30") int limit,
   @RequestParam(required = false) String sort,
   @RequestParam(required = false) String dir) {
    PageBounds pageBounds = new PageBounds(page, limit, Order.create(sort, dir));
    pageBounds.setAsyncTotalCount(true);
   return loginService.getAllUsersByPage(pageBounds);
  }
你好,请问我的返回结果为什么没有 totalCount 呢?

[{"userid":7,"username":"user6","password":"pass2","enabled":1,"issys":1,"user_desc":"1"},{"userid":6,"username":"user5","password":"pass2","enabled":1,"issys":1,"user_desc":"1"},{"userid":5,"username":"user4","password":"pass2","enabled":1,"issys":1,"user_desc":"1"}]

spring mvc的objectMapper配置了没,上面有配置方法
ponta

引用来自“miemiedev”的评论

引用来自“ponta”的评论

  @ResponseBody
  @RequestMapping(value = "/getAllUsersByPage.json",method=RequestMethod.GET)
  public List<Users> getAllUsersByPage(
   @RequestParam(required = false,defaultValue = "1") int page,
   @RequestParam(required = false,defaultValue = "30") int limit,
   @RequestParam(required = false) String sort,
   @RequestParam(required = false) String dir) {
    PageBounds pageBounds = new PageBounds(page, limit, Order.create(sort, dir));
    pageBounds.setAsyncTotalCount(true);
   return loginService.getAllUsersByPage(pageBounds);
  }
你好,请问我的返回结果为什么没有 totalCount 呢?

[{"userid":7,"username":"user6","password":"pass2","enabled":1,"issys":1,"user_desc":"1"},{"userid":6,"username":"user5","password":"pass2","enabled":1,"issys":1,"user_desc":"1"},{"userid":5,"username":"user4","password":"pass2","enabled":1,"issys":1,"user_desc":"1"}]

spring mvc的objectMapper配置了没,上面有配置方法

  <mvc:annotation-driven>
  <mvc:message-converters register-defaults="true">
  <bean class="org.springframework.http.converter.StringHttpMessageConverter">
  <constructor-arg value="UTF-8" />
  </bean>
  <bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter">
  <property name="objectMapper">
    <bean class="com.github.miemiedev.mybatis.paginator.jackson2.PageListJsonMapper" />
  </property>
  </bean>
  </mvc:message-converters>
  </mvc:annotation-driven>

是这个吗?我直接复制的你的啊。
ponta

引用来自“miemiedev”的评论

引用来自“ponta”的评论

  @ResponseBody
  @RequestMapping(value = "/getAllUsersByPage.json",method=RequestMethod.GET)
  public List<Users> getAllUsersByPage(
   @RequestParam(required = false,defaultValue = "1") int page,
   @RequestParam(required = false,defaultValue = "30") int limit,
   @RequestParam(required = false) String sort,
   @RequestParam(required = false) String dir) {
    PageBounds pageBounds = new PageBounds(page, limit, Order.create(sort, dir));
    pageBounds.setAsyncTotalCount(true);
   return loginService.getAllUsersByPage(pageBounds);
  }
你好,请问我的返回结果为什么没有 totalCount 呢?

[{"userid":7,"username":"user6","password":"pass2","enabled":1,"issys":1,"user_desc":"1"},{"userid":6,"username":"user5","password":"pass2","enabled":1,"issys":1,"user_desc":"1"},{"userid":5,"username":"user4","password":"pass2","enabled":1,"issys":1,"user_desc":"1"}]

spring mvc的objectMapper配置了没,上面有配置方法

可以了,谢谢!,原因是我配置的前面 已经有<mvc:annotation-driven /> 了,把这行注释掉就行了。原因不明!
飞天小色猫
没有把结果集在封装下 totalPage里面么?
×
miemiedev
如果觉得我的文章对您有用,请随意打赏。您的支持将鼓励我继续创作!
* 金额(元)
¥1 ¥5 ¥10 ¥20 其他金额
打赏人
留言
* 支付类型
微信扫码支付
打赏金额:
已支付成功
打赏金额: