- Notifications
You must be signed in to change notification settings - Fork34
bus-pager
Folders and files
Name | Name | Last commit message | Last commit date | |
---|---|---|---|---|
parent directory.. | ||||
该插件目前支持以下数据库的物理分页:
Oracle
Mysql
MariaDB
SQLite
Hsqldb
PostgreSQL
DB2
SqlServer(2005,2008)
Informix
H2
SqlServer2012
Derby
Phoenix
- 达梦数据库(dm)
- 阿里云PPAS数据库
引入分页插件有下面2种方式,推荐使用 Maven 方式
你可以从下面的地址中下载最新版本的 jar 包
由于使用了sql 解析工具,你还需要下载 jsqlparser.jar(需要和bus-pager依赖的版本一致) :
在 pom.xml 中添加如下依赖:
<dependency> <groupId>org.miaixz</groupId> <artifactId>bus-pager</artifactId> <version>x.x.x</version></dependency>
最新版本号可以从首页查看。
特别注意,新版拦截器是org.miaixz.bus.pager.plugin.PageInterceptor
。
<beanid="sqlSessionFactory"class="org.mybatis.spring.SqlSessionFactoryBean"><!-- 注意其他配置--> <propertyname="plugins"> <array> <beanclass="org.miaixz.bus.pager.plugin.PageInterceptor"> <propertyname="properties"><!--使用下面的方式配置参数,一行配置一个--> <value> params=value1 </value> </property> </bean> </array> </property></bean>
分页插件提供了多个可选参数,这些参数使用时,按照上面两种配置方式中的示例配置即可。
分页插件可选参数如下:
dialect
:默认情况下会使用 Page方式进行分页,如果想要实现自己的分页逻辑,可以实现Dialect
(org.miaixz.bus.pager.dialect.Dialect
)接口,然后配置该属性为实现类的全限定名称。
下面几个参数都是针对默认 dialect 情况下的参数。使用自定义 dialect 实现时,下面的参数没有任何作用。
delegate
:分页插件会自动检测当前的数据库链接,自动选择合适的分页方式。 你可以配置delegate
属性来指定分页插件使用哪种方言。配置时,可以使用下面的缩写值:oracle
,mysql
,mariadb
,sqlite
,hsqldb
,postgresql
,db2
,sqlserver
,informix
,h2
,sqlserver2012
,derby
特别注意:使用 SqlServer2012 数据库时,需要手动指定为sqlserver2012
,否则会使用 SqlServer2005 的方式进行分页。你也可以实现AbstractPaging
,然后配置该属性为实现类的全限定名称即可使用自定义的实现方法。offsetAsPageNo
:默认值为false
,该参数对使用RowBounds
作为分页参数时有效。 当该参数设置为true
时,会将RowBounds
中的offset
参数当成pageNo
使用,可以用页码和页面大小两个参数进行分页。rowBoundsWithCount
:默认值为false
,该参数对使用RowBounds
作为分页参数时有效。 当该参数设置为true
时,使用RowBounds
分页会进行 count 查询。pageSizeZero
:默认值为false
,当该参数设置为true
时,如果pageSize=0
或者RowBounds.limit = 0
就会查询出全部的结果(相当于没有执行分页查询,但是返回结果仍然是Page
类型)。reasonable
:分页合理化参数,默认值为false
。当该参数设置为true
时,pageNo<=0
时会查询第一页,pageNo>pages
(超过总数时),会查询最后一页。默认false
时,直接根据参数进行查询。params
:为了支持startPage(Object params)
方法,增加了该参数来配置参数映射,用于从对象中根据属性名取值,可以配置pageNo,pageSize,count,pageSizeZero,reasonable
,不配置映射的用默认值,默认值为pageNo=pageNo;pageSize=pageSize;count=countSql;reasonable=reasonable;pageSizeZero=pageSizeZero
。supportMethodsArguments
:支持通过 Mapper 接口参数来传递分页参数,默认值false
,分页插件会从查询方法的参数值中,自动根据上面params
配置的字段中取值,查找到合适的值时就会自动分页。使用方法可以参考测试代码中的org.miaixz.bus.pager.test.basic
包下的ArgumentsMapTest
和ArgumentsObjTest
。autoRuntimeDialect
:默认值为false
。设置为true
时,允许在运行时根据多数据源自动识别对应方言的分页(不支持自动选择sqlserver2012
,只能使用sqlserver
),用法和注意事项参考下面的场景五。closeConn
:默认值为true
。当使用运行时动态数据源或没有设置delegate
属性自动获取数据库类型时,会自动获取一个数据库连接,通过该属性来设置是否关闭获取的这个连接,默认true
关闭,设置为false
后,不会关闭获取的连接,这个参数的设置要根据自己选择的数据源来决定。aggregateFunctions
(5.1.5+):默认为所有常见数据库的聚合函数,允许手动添加聚合函数(影响行数),所有以聚合函数开头的函数,在进行count 转换时,会套一层。其他函数和列会被替换为 count(0),其中count列可以自己配置。
重要提示:
当offsetAsPageNo=false
的时候,由于pageNo
问题,RowBounds
查询的时候reasonable
会强制为false
。使用PageContext.startPage
方法不受影响。
单独看每个参数的说明可能是一件让人不爽的事情,这里列举一些可能会用到某些参数的情况。
如果你仍然在用类似ibatis式的命名空间调用方式,你也许会用到rowBoundsWithCount
, 分页插件对RowBounds
支持和 MyBatis默认的方式是一致,默认情况下不会进行 count查询,如果你想在分页查询时进行 count 查询, 以及使用更强大的Page
类,你需要设置该参数为true
。
注:PageRowBounds
想要查询总数也需要配置该属性为true
。
如果你仍然在用类似ibatis式的命名空间调用方式,你觉得RowBounds
中的两个参数offset,limit
不如PageNo,pageSize
容易理解,你可以使用offsetAsPageNo
参数,将该参数设置为true
后,offset
会当成pageNo
使用,limit
和pageSize
含义相同。
如果觉得某个地方使用分页后,你仍然想通过控制参数查询全部的结果,你可以配置pageSizeZero
为true
,配置后,当pageSize=0
或者RowBounds.limit = 0
就会查询出全部的结果。
如果你分页插件使用于类似分页查看列表式的数据,如新闻列表,软件列表, 你希望用户输入的页数不在合法范围(第一页到最后一页之外)时能够正确的响应到正确的结果页面,那么你可以配置reasonable
为true
,这时如果PageNo<=0
会查询第一页,如果PageNo>总页数
会查询最后一页。
如果你在 Spring 中配置了动态数据源,并且连接不同类型的数据库,这时你可以配置autoRuntimeDialect
为true
,这样在使用不同数据源时,会使用匹配的分页进行查询。这种情况下,你还需要特别注意closeConn
参数,由于获取数据源类型会获取一个数据库连接,所以需要通过这个参数来控制获取连接后,是否关闭该连接。默认为true
,有些数据库连接关闭后就没法进行后续的数据库操作。而有些数据库连接不关闭就会很快由于连接数用完而导致数据库无响应。所以在使用该功能时,特别需要注意你使用的数据源是否需要关闭数据库连接。
当不使用动态数据源而只是自动获取dialect
时,数据库连接只会获取一次,所以不需要担心占用的这一个连接是否会导致数据库出错,但是最好也根据数据源的特性选择是否关闭连接。
分页插件支持以下几种调用方式:
//第一种,RowBounds方式的调用List<User>list =sqlSession.selectList("x.y.selectIf",null,newRowBounds(0,10));//第二种,Mapper接口方式的调用,推荐这种使用方式。PageContext.startPage(1,10);List<User>list =userMapper.selectIf(1);//第三种,Mapper接口方式的调用,推荐这种使用方式。PageContext.offsetPage(1,10);List<User>list =userMapper.selectIf(1);//第四种,参数方法调用//存在以下 Mapper 接口方法,你不需要在 xml 处理后两个参数publicinterfaceCountryMapper {List<User>selectByPageNoSize(@Param("user")Useruser,@Param("PageNo")intPageNo,@Param("pageSize")intpageSize);}//配置supportMethodsArguments=true//在代码中直接调用:List<User>list =userMapper.selectByPageNoSize(user,1,10);//第五种,参数对象//如果 PageNo 和 pageSize 存在于 User 对象中,只要参数有值,也会被分页//有如下 User 对象publicclassUser {//其他fields//下面两个参数名和 params 配置的名字一致privateIntegerPageNo;privateIntegerpageSize;}//存在以下 Mapper 接口方法,你不需要在 xml 处理后两个参数publicinterfaceCountryMapper {List<User>selectByPageNoSize(Useruser);}//当 user 中的 null != PageNo && null != pageSize 时,会自动分页List<User>list =userMapper.selectByPageNoSize(user);//第六种,Querying 接口方式//jdk6,7用法,创建接口Page<User>page =PageContext.startPage(1,10).doSelectPage(newISelect() {@OverridepublicvoiddoSelect() {userMapper.selectGroupBy(); }});//jdk8 lambda用法Page<User>page =PageContext.startPage(1,10).doSelectPage(() ->userMapper.selectGroupBy());//也可以直接返回Page,注意doSelectPage方法和doSelectPagepage=PageContext.startPage(1,10).doSelectPage(newISelect() {@OverridepublicvoiddoSelect () {userMapper.selectGroupBy(); }});//对应的lambda用法Page=PageContext.startPage(1,10).doSelectPage(()->userMapper.selectGroupBy());//count查询,返回一个查询语句的count数longtotal =PageContext.count(newISelect() {@OverridepublicvoiddoSelect() {userMapper.selectLike(user); }});//lambdatotal=PageContext.count(()->userMapper.selectLike(user));
下面对最常用的方式进行详细介绍
List<User>list=sqlSession.selectList("x.y.selectIf",null,newRowBounds(1,10));
使用这种调用方式时,你可以使用RowBounds参数进行分页,这种方式侵入性最小,我们可以看到,通过RowBounds方式调用只是使用了这个参数,并没有增加其他任何内容。
分页插件检测到使用了RowBounds参数时,就会对该查询进行物理分页。
关于这种方式的调用,有两个特殊的参数是针对RowBounds
的,你可以参看上面的场景一 和场景二
注:不只有命名空间方式可以用RowBounds,使用接口的时候也可以增加RowBounds参数,例如:
//这种情况下也会进行物理分页查询List<User>selectAll(RowBoundsrowBounds);
注意: 由于默认情况下的RowBounds
无法获取查询总数,分页插件提供了一个继承自RowBounds
的PageRowBounds
,这个对象中增加了total
属性,执行分页查询后,可以从该属性得到查询总数。
除了PageContext.startPage
方法外,还提供了类似用法的PageContext.offsetPage
方法。
在你需要进行分页的 MyBatis 查询方法前调用PageContext.startPage
静态方法即可,紧跟在这个方法后的第一个**MyBatis 查询方法**会被进行分页。
//获取第1页,10条内容,默认查询总数countPageContext.startPage(1,10);//紧跟着的第一个select方法会被分页List<User>list=userMapper.selectIf(1);assertEquals(2,list.get(0).getId());assertEquals(10,list.size());//分页时,实际返回的结果list类型是Page<E>,如果想取出分页信息,需要强制转换为Page<E>assertEquals(182,((Page)list).getTotal());
//request: url?PageNo=1&pageSize=10//支持 ServletRequest,Map,POJO 对象,需要配合 params 参数PageContext.startPage(request);//紧跟着的第一个select方法会被分页List<User>list =userMapper.selectIf(1);//后面的不会被分页,除非再次调用PageContext.startPageList<User>list2 =userMapper.selectIf(null);//list1assertEquals(2,list.get(0).getId());assertEquals(10,list.size());//分页时,实际返回的结果list类型是Page<E>,如果想取出分页信息,需要强制转换为Page<E>,//或者使用Page类(下面的示例有介绍)assertEquals(182,((Page)list).getTotal());//list2assertEquals(1,list2.get(0).getId());assertEquals(182,list2.size());
//获取第1页,10条内容,默认查询总数countPageContext.startPage(1,10);List<User>list=userMapper.selectAll();//用Page对结果进行包装Pagepage=newPage(list);//测试Page全部属性//Page包含了非常全面的分页属性assertEquals(1,page.getPageNo());assertEquals(10,page.getPageSize());assertEquals(1,page.getStartRow());assertEquals(10,page.getEndRow());assertEquals(183,page.getTotal());assertEquals(19,page.getPages());assertEquals(1,page.getFirstPage());assertEquals(8,page.getLastPage());assertEquals(true,page.isFirstPage());assertEquals(false,page.isLastPage());assertEquals(false,page.isHasPreviousPage());assertEquals(true,page.isHasNextPage());
想要使用参数方式,需要配置supportMethodsArguments
参数为true
,同时要配置params
参数。 例如下面的配置:
<plugins><!-- org.miaixz.bus.pager为PageContext类所在包名--> <plugininterceptor="org.miaixz.bus.pager.plugin.PageInterceptor"><!-- 使用下面的方式配置参数,后面会有所有的参数介绍--> <propertyname="supportMethodsArguments"value="true"/> <propertyname="params"value="PageNo=PageNoKey;pageSize=pageSizeKey;"/> </plugin></plugins>
在 MyBatis 方法中:
List<User>selectByPageNoSize(@Param("user")Useruser,@Param("PageNoKey")intPageNo,@Param("pageSizeKey")intpageSize);
当调用这个方法时,由于同时发现了PageNoKey
和pageSizeKey
参数,这个方法就会被分页。params 提供的几个参数都可以这样使用。
除了上面这种方式外,如果 User 对象中包含这两个参数值,也可以有下面的方法:
List<User>selectByPageNoSize(Useruser);
当从 User 中同时发现了PageNoKey
和pageSizeKey
参数,这个方法就会被分页。
注意:pageNo
和pageSize
两个属性同时存在才会触发分页操作,在这个前提下,其他的分页参数才会生效。
ISelect 接口方式除了可以保证安全外,还特别实现了将查询转换为单纯的 count查询方式,这个方法可以将任意的查询方法,变成一个select count(*)
的查询方法。
PageContext
方法使用了静态的ThreadLocal
参数,分页参数和线程是绑定的。
只要你可以保证在PageContext
方法调用后紧跟 MyBatis 查询方法,这就是安全的。因为PageContext
在finally
代码段中自动清除了ThreadLocal
存储的对象。
如果代码在进入Executor
前发生异常,就会导致线程不可用,这属于人为的 Bug(例如接口方法和 XML中的不匹配,导致找不到MappedStatement
时),这种情况由于线程不可用,也不会导致ThreadLocal
参数被错误的使用。
但是如果你写出下面这样的代码,就是不安全的用法:
PageContext.startPage(1,10);List<User>list;if(param1!=null){list=userMapper.selectIf(param1); }else{list=newArrayList<User>(); }
这种情况下由于 param1 存在 null 的情况,就会导致 PageContext生产了一个分页参数,但是没有被消费,这个参数就会一直保留在这个线程上。当这个线程再次被使用时,就可能导致不该分页的方法去消费这个分页参数,这就产生了莫名其妙的分页。
上面这个代码,应该写成下面这个样子:
List<User>list;if(param1!=null){PageContext.startPage(1,10);list=userMapper.selectIf(param1); }else{list=newArrayList<User>(); }
这种写法就能保证安全。
如果你对此不放心,你可以手动清理ThreadLocal
存储的分页参数,可以像下面这样使用:
List<User>list;if(param1!=null){PageContext.startPage(1,10);try{list=userMapper.selectAll(); }finally{PageContext.clearPage(); } }else{list=newArrayList<User>(); }
这么写很不好看,而且没有必要