PageHelper和Mybatis Generator的入门使用

Mybatis Generator 可以根据数据库中的对应的表(具体根据哪张表需要在generationConfig.xml文件中配置,配置信息可以查看上方连接)生成对应的实体类,dao接口以及单表查询接口对应的mapper.xml文件(也可以选择用注解的方式生成)。

PageHelper 是一个分页插件,可以很方便的进行分页查询。

项目结构: https://github.com/abel533/MyBatis-Spring-Boot(参考文件地址)
请添加图片描述

1、Maven核心依赖

<!--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>

2、application.yml配置

spring:
  datasource:
    password: 123456
    username: root
    url: jdbc:mysql://localhost:3306/test?useUnicode=true&characterEncoding=utf-8&useSSL=true&serverTimezone=UTC
    driver-class-name: com.mysql.cj.jdbc.Driver

mybatis:
  mapper-locations: classpath:mapper/*.xml
  type-aliases-package: com.zhikan.pagehelperdemo.entity

mapper:
  mappers:
    - com.zhikan.pagehelperdemo.mapper
  not-empty: false
  identity: MYSQL

pagehelper:
  dialect: mysql
  reasonable: true
  support-methods-arguments: true
  params: countsql

3、generatorConfig.xml配置(根据文件进行配置)

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE generatorConfiguration
        PUBLIC "-//mybatis.org//DTD MyBatis Generator Configuration 1.0//EN"
        "http://mybatis.org/dtd/mybatis-generator-config_1_0.dtd">

<generatorConfiguration>
    <properties resource="application-dev.properties"/>

    <context id="Mysql" targetRuntime="MyBatis3Simple" defaultModelType="flat">
        <property name="beginningDelimiter" value="`"/>
        <property name="endingDelimiter" value="`"/>

        <plugin type="tk.mybatis.mapper.generator.MapperPlugin">
            <property name="mappers" value="tk.mybatis.springboot.util.MyMapper"/>
        </plugin>

        <jdbcConnection driverClass="${spring.datasource.driver-class-name}"
                        connectionURL="${spring.datasource.url}"
                        userId="${spring.datasource.username}"
                        password="${spring.datasource.password}">
        </jdbcConnection>

        <javaModelGenerator targetPackage="tk.mybatis.springboot.model" targetProject="src/main/java"/>

        <sqlMapGenerator targetPackage="mapper" targetProject="src/main/resources"/>

        <javaClientGenerator targetPackage="tk.mybatis.springboot.mapper" targetProject="src/main/java"
                             type="XMLMAPPER"/>

        <table tableName="country">
            <!--mysql 配置-->
            <generatedKey column="id" sqlStatement="Mysql" identity="true"/>
            <!--oracle 配置-->
            <!--<generatedKey column="id" sqlStatement="select SEQ_{1}.nextval from dual" identity="false" type="pre"/>-->
        </table>
    </context>
</generatorConfiguration>

4、生成对应的实体类、DAO和mapper.xml文件

在IDEA中打开终端:

  • set MAVEN_OPTS="-Dfile.encoding=UTF-8"(每次执行下面命令时,先执行该命令,否则生成的实体类的中文注释乱码)

  • set MAVEN_OPTS="-Dfile.encoding=UTF-8"(执行该命令,对应目录下生成文件)

5、分页插件的使用

在导入相应的maven依赖后,可以使用PageHelper.startPage(pageNum,pageSize)方法来设置页数和页数大小,然后通过DAO层获取查询结果,最后结果使用PageInfo进行封装,最后返回给前端。

    @Override
    public ApiResult<PageInfo<UserManagerVO>> search(UserManagerSearchDTO dto) {
        List<UserManagerVO> list = null;
        PageHelper.startPage(dto.getPageNumber(),dto.getPageSize());
        list = userManagerMapper.queryBySearchValue(dto);
        PageInfo<UserManagerVO> userManagerVOPageInfo = new PageInfo<>(list);
        return ApiResult.success(userManagerVOPageInfo);
    }

6、PageHelper方法分页的原理:

    - PageHelper方法使用了静态的ThreadLocal参数,分页参数和线程是绑定的 。内部流程ThreadLocal中设置了分页参数(pageNum,pageSize),之后在查询执行的时候,获取当线程中的分页参数,执行查询的时候通过拦截器在sql语句中添加分页参数,之后实现分页查询,查询结束后在 finally 语句中清除ThreadLocal中的查询参数

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-XeKNbHvu-1627376558312)(image/image_1.png)]

    详细时序图

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Va3gWMFs-1627376558313)(image/image_2.png)]

    - 注意:只要你可以保证在PageHelper方法调用后紧跟 MyBatis 查询方法,这就是安全的。因为PageHelper在finally代码段中自动清除了ThreadLocal存储的对象。如果代码进入Executor前发生异常,就会导致线程不可用,这属于认为bug(例如接口方法和xml文件的方法不匹配,导致找不mappedStatement时),这种情况由于线程不可用,也不会导致ThreadLocal参数被错误使用。

    - 但是下面这种写法存在问题:

PageHelper.StartPage(1, 10);
List<Country> list;
if (param1 != null){
    list = countryMapper.selectIf(param1);
} else {
    list = new ArrayList<Country>();
}

        这种情况下,当param1 == null 时,会导致PageHelper产生一个分页,但是这个分页并没有被消费,这个参数就会一直保留在这个线程上。当这个线程再次被使用时,可能会导致本来不分页的操作查出来分页的结果。应该使用下面这种写法保证线程安全。

List<Country> list;
if (param1 != null){
    PageHelper.StartPage(1, 10);
    list = countryMapper.selectIf(param1);
} else {
    list = new ArrayList<Country>();
}