Dukou007 发表于 2022-9-7 11:05

Pagehelper使用教程

## Pagehelper使用教程

### 原理

`PageHelper` 方法使用了静态的 `ThreadLocal` 参数,分页参数和线程是绑定的。

只要你可以保证在 `PageHelper` 方法调用后紧跟 MyBatis 查询方法,这就是安全的。因为 `PageHelper` 在 `finally` 代码段中自动清除了 `ThreadLocal` 存储的对象。

如果代码在进入 `Executor` 前发生异常,就会导致线程不可用,这属于人为的 Bug(例如接口方法和 XML 中的不匹配,导致找不到 `MappedStatement` 时), 这种情况由于线程不可用,也不会导致 `ThreadLocal` 参数被错误的使用。

#### 参数介绍

1. `helperDialect`:分页插件会自动检测当前的数据库链接,自动选择合适的分页方式。 你可以配置`helperDialect`属性来指定分页插件使用哪种方言。配置时,可以使用下面的缩写值:
   `oracle`,`mysql`,`mariadb`,`sqlite`,`hsqldb`,`postgresql`,`db2`,`sqlserver`,`informix`,`h2`,`sqlserver2012`,`derby`
   **特别注意:**使用 SqlServer2012 数据库时,需要手动指定为 `sqlserver2012`,否则会使用 SqlServer2005 的方式进行分页。
   你也可以实现 `AbstractHelperDialect`,然后配置该属性为实现类的全限定名称即可使用自定义的实现方法。
2. `offsetAsPageNum`:默认值为 `false`,该参数对使用 `RowBounds` 作为分页参数时有效。 当该参数设置为 `true` 时,会将 `RowBounds` 中的 `offset` 参数当成 `pageNum` 使用,可以用页码和页面大小两个参数进行分页。
3. `rowBoundsWithCount`:默认值为`false`,该参数对使用 `RowBounds` 作为分页参数时有效。 当该参数设置为`true`时,使用 `RowBounds` 分页会进行 count 查询。
4. `pageSizeZero`:默认值为 `false`,当该参数设置为 `true` 时,如果 `pageSize=0` 或者 `RowBounds.limit = 0` 就会查询出全部的结果(相当于没有执行分页查询,但是返回结果仍然是 `Page` 类型)。
5. `reasonable`:分页合理化参数,默认值为`false`。当该参数设置为 `true` 时,`pageNum<=0` 时会查询第一页, `pageNum>pages`(超过总数时),会查询最后一页。默认`false` 时,直接根据参数进行查询。
6. `params`:为了支持`startPage(Object params)`方法,增加了该参数来配置参数映射,用于从对象中根据属性名取值, 可以配置 `pageNum,pageSize,count,pageSizeZero,reasonable`,不配置映射的用默认值, 默认值为`pageNum=pageNum;pageSize=pageSize;count=countSql;reasonable=reasonable;pageSizeZero=pageSizeZero`。
7. `supportMethodsArguments`:支持通过 Mapper 接口参数来传递分页参数,默认值`false`,分页插件会从查询方法的参数值中,自动根据上面 `params` 配置的字段中取值,查找到合适的值时就会自动分页。 使用方法可以参考测试代码中的 `com.github.pagehelper.test.basic` 包下的 `ArgumentsMapTest` 和 `ArgumentsObjTest`。
8. `autoRuntimeDialect`:默认值为 `false`。设置为 `true` 时,允许在运行时根据多数据源自动识别对应方言的分页 (不支持自动选择`sqlserver2012`,只能使用`sqlserver`),用法和注意事项参考下面的**场景五**。
9. `closeConn`:默认值为 `true`。当使用运行时动态数据源或没有设置 `helperDialect` 属性自动获取数据库类型时,会自动获取一个数据库连接, 通过该属性来设置是否关闭获取的这个连接,默认`true`关闭,设置为 `false` 后,不会关闭获取的连接,这个参数的设置要根据自己选择的数据源来决定。
10. `aggregateFunctions`(5.1.5+):默认为所有常见数据库的聚合函数,允许手动添加聚合函数(影响行数),所有以聚合函数开头的函数,在进行 count 转换时,会套一层。其他函数和列会被替换为 count(0),其中count列可以自己配置。

**重要提示:**

当 `offsetAsPageNum=false` 的时候,由于 `PageNum` 问题,`RowBounds`查询的时候 `reasonable` 会强制为 `false`。使用 `PageHelper.startPage` 方法不受影响。

### 准备工作

#### maven中

```xml
                <!--mybatis-->
                <dependency>
                  <groupId>org.mybatis.spring.boot</groupId>
                  <artifactId>mybatis-spring-boot-starter</artifactId>
                  <version>1.3.1</version>
                </dependency>
                <!--mapper-->
                <dependency>
                  <groupId>tk.mybatis</groupId>
                  <artifactId>mapper-spring-boot-starter</artifactId>
                  <version>1.2.4</version>
                </dependency>
                <!--pagehelper-->
                <dependency>
                  <groupId>com.github.pagehelper</groupId>
                  <artifactId>pagehelper-spring-boot-starter</artifactId>
                  <version>1.2.3</version>
                </dependency>
```

#### 配置拦截器插件

特别注意,新版拦截器是 `com.github.pagehelper.PageInterceptor`。 `com.github.pagehelper.PageHelper` 现在是一个特殊的 `dialect` 实现类,是分页插件的默认实现类,提供了和以前相同的用法。

##### 在 MyBatis 配置 xml 中配置拦截器插件

```
<!--
    plugins在配置文件中的位置必须符合要求,否则会报错,顺序如下:
    properties?, settings?,
    typeAliases?, typeHandlers?,
    objectFactory?,objectWrapperFactory?,
    plugins?,
    environments?, databaseIdProvider?, mappers?
-->
<plugins>
    <!-- com.github.pagehelper为PageHelper类所在包名 -->
    <plugin interceptor="com.github.pagehelper.PageInterceptor">
      <!-- 使用下面的方式配置参数,后面会有所有的参数介绍 -->
      <property name="param1" value="value1"/>
        </plugin>
</plugins>
```

##### 在 Spring 配置文件中配置拦截器插件

使用 spring 的属性配置方式,可以使用 `plugins` 属性像下面这样配置:

```xml
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<!-- 注意其他配置 -->
<property name="plugins">
    <array>
      <bean class="com.github.pagehelper.PageInterceptor">
      <property name="properties">
          <!--使用下面的方式配置参数,一行配置一个 -->
          <value>
            params=value1
          </value>
      </property>
      </bean>
    </array>
</property>
</bean>
```

#### application.properties

```
#mybatis
mybatis.type-aliases-package=tk.mybatis.springboot.model
mybatis.mapper-locations=classpath:mapper/*.xml

#mapper
#mappers 多个接口时逗号隔开
mapper.mappers=tk.mybatis.springboot.util.MyMapper
mapper.not-empty=false
mapper.identity=MYSQL

#pagehelper
pagehelper.helperDialect=mysql
pagehelper.reasonable=true
pagehelper.supportMethodsArguments=true
pagehelper.params=count=countSql

```

#### application.yml

```xml
                mybatis:
                  type-aliases-package: tk.mybatis.springboot.model
                  mapper-locations: classpath:mapper/*.xml
               
                mapper:
                  mappers:
                        - tk.mybatis.springboot.util.MyMapper
                  not-empty: false
                  identity: MYSQL
               
                pagehelper:
               helperDialect: mysql #数据库方言
               reasonable: true # 分页合理化参数,默认值为false。当该参数设置为 true 时,pageNum<=0 时会查询第一页,pageNum>pages(超过总数时),会查询最后一页。默认false 时,直接根据参数进行查询。
               supportMethodsArguments: true # 支持通过 Mapper 接口参数来传递分页参
               params: count=countSql # 增加了该参数来配置参数映射,用于从对象中根据属性名取值
```

### 使用过程

##### 第一种,RowBounds方式的调用

```java
List<User> list = sqlSession.selectList("x.y.selectIf", null, new RowBounds(0, 10));
```

##### 第二种,Mapper接口方式的调用,推荐这种使用方式。

```java
PageHelper.startPage(1, 10);
List<User> list = userMapper.selectIf(1);
```

##### 第三种,Mapper接口方式的调用,推荐这种使用方式。

```java
PageHelper.offsetPage(1, 10);
List<User> list = userMapper.selectIf(1);
```

##### 第四种,参数方法调用

```java
# 存在以下 Mapper 接口方法,你不需要在 xml 处理后两个参数
public interface CountryMapper {
    List<User> selectByPageNumSize(
            @Param("user") User user,
            @Param("pageNum") int pageNum,
            @Param("pageSize") int pageSize);
}
#配置supportMethodsArguments=true,在代码中直接调用:
List<User> list = userMapper.selectByPageNumSize(user, 1, 10);
```

##### 第五种,参数对象

```java
### 如果 pageNum 和 pageSize 存在于 User 对象中,只要参数有值,也会被分页 .有如下 User 对象
public class User {
    其他fields
    下面两个参数名和 params 配置的名字一致
    private Integer pageNum;
    private Integer pageSize;
}
### 存在以下 Mapper 接口方法,你不需要在 xml 处理后两个参数
public interface CountryMapper {
    List<User> selectByPageNumSize(User user);
}
### 当 user 中的 pageNum!= null && pageSize!= null 时,会自动分页
List<User> list = userMapper.selectByPageNumSize(user);
```

##### 第六种,ISelect 接口方式

```java
jdk6,7用法,创建接口
Page<User> page = PageHelper.startPage(1, 10).doSelectPage(new ISelect() {
    @Override
    public void doSelect() {
      userMapper.selectGroupBy();
    }
});
jdk8 lambda用法

Page<User> page = PageHelper.startPage(1, 10).doSelectPage(()-> userMapper.selectGroupBy());

也可以直接返回PageInfo,注意doSelectPageInfo方法和doSelectPage
pageInfo = PageHelper.startPage(1, 10).doSelectPageInfo(new ISelect() {
    @Override
    public void doSelect() {
      userMapper.selectGroupBy();
    }
});
对应的lambda用法
pageInfo = PageHelper.startPage(1, 10).doSelectPageInfo(() -> userMapper.selectGroupBy());

count查询,返回一个查询语句的count数
long total = PageHelper.count(new ISelect() {
    @Override
    public void doSelect() {
      userMapper.selectLike(user);
    }
});
lambda
total = PageHelper.count(()->userMapper.selectLike(user));
```

### 可能存在的问题及解决方案

##### Spring DevTools 配置

现象:在使用 DevTools 时,通用Mapper经常会出现 class x.x.A cannot be cast to x.x.A。

分析:同一个类如果使用了不同的类加载器,就会产生这样的错误,所以解决方案就是让通用Mapper和实体类使用相同的类加载器即可。DevTools 默认会对 IDE 中引入的所有项目使用 restart 类加载器,对于引入的 jar 包使用 base 类加载器,因此只要保证通用Mapper的jar包使用 restart 类加载器即可。

解决:

在 `src/main/resources` 中创建 META-INF 目录,在此目录下添加 spring-devtools.properties 配置,使用这个配置后,就会使用 restart 类加载加载 include 进去的 jar 包。内容如下:

```xml
restart.include.mapper=/mapper-[\\w-\\.]+jar
restart.include.pagehelper=/pagehelper-[\\w-\\.]+jar
```

##### 不安全的分页

如果代码在进入 `Executor` 前发生异常,就会导致线程不可用,这属于人为的 Bug(例如接口方法和 XML 中的不匹配,导致找不到 `MappedStatement` 时), 这种情况由于线程不可用,也不会导致 `ThreadLocal` 参数被错误的使用。

```java
PageHelper.startPage(1, 10);
List<User> list;
if(param1 != null){
    list = userMapper.selectIf(param1);
} else {
    list = new ArrayList<User>();
}
```

这种情况下由于 param1 存在 null 的情况,就会导致 PageHelper 生产了一个分页参数,但是没有被消费,这个参数就会一直保留在这个线程上。当这个线程再次被使用时,就可能导致不该分页的方法去消费这个分页参数,这就产生了莫名其妙的分页。

正确方式:

```java
List<User> list;
if(param1 != null){
    PageHelper.startPage(1, 10);
    try{
      list = userMapper.selectAll();
    } finally {
      PageHelper.clearPage();
    }
} else {
    list = new ArrayList<User>();
}
```



### 参考地址:

https://github.com/pagehelper/Mybatis-PageHelper/blob/master/wikis/en/HowToUse.md
页: [1]
查看完整版本: Pagehelper使用教程