zoukankan      html  css  js  c++  java
  • 通用Mapper与分页插件的集成

    SpringBoot 是为了简化 Spring 应用的创建、运行、调试、部署等一系列问题而诞生的产物,自动装配的特性让我们可以更好的关注业务本身而不是外部的XML配置,我们只需遵循规范,引入相关的依赖就可以轻易的搭建出一个 WEB 工程

    一起来学SpringBoot | 第七篇:整合Mybatis一文中,我们介绍了Mybatis这款优秀的框架,顺便提及了民间大神开发的两款插件通用MapperPageHelper,从此告别简单CURD代码的编写….

    插件介绍

    以下两款插件作者均是同一个人,如果你想深入了解Mybatis以及插件开发可以购买作者的书籍

    MyBatis 从入门到精通MyBatis 从入门到精通

    分页插件

    在没有分页插件之前,写一个分页需要两条SQL语句,一条查询一条统计,然后才能计算出页码,这样的代码冗余而又枯燥,更重要的一点是数据库迁移,众所周知不同的数据库分页写法是不同的,而Mybatis不同于Hibernate的是它只提供动态SQL和结果集映射。值得庆幸的是,它虽然没有为分页提供良好的解决方案,但却提供了Interceptor以供开发者自己扩展,这也是这款分页插件的由来….

    通用Mapper

    通用 Mapper 是一个可以实现任意 MyBatis 通用方法的框架,项目提供了常规的增删改查操作以及 Example 相关的单表操作。通用 Mapper 是为了解决 MyBatis 使用中 90% 的基本操作,使用它可以很方便的进行开发,可以节省开发人员大量的时间。

    导入依赖

    在 pom.xml 中添加通用Mapper与分页插件的依赖包

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    <!-- 通用Mapper插件
    文档地址:https://gitee.com/free/Mapper/wikis/Home -->
    <dependency>
    <groupId>tk.mybatis</groupId>
    <artifactId>mapper-spring-boot-starter</artifactId>
    <version>2.0.2</version>
    </dependency>
    <!-- 分页插件
    文档地址:https://github.com/pagehelper/Mybatis-PageHelper/blob/master/wikis/zh/HowToUse.md -->
    <dependency>
    <groupId>com.github.pagehelper</groupId>
    <artifactId>pagehelper-spring-boot-starter</artifactId>
    <version>1.2.5</version>
    </dependency>
    <!-- MYSQL包 -->
    <dependency>
    <groupId>mysql</groupId>
    <artifactId>mysql-connector-java</artifactId>
    </dependency>
    <!-- 默认就内嵌了Tomcat 容器,如需要更换容器也极其简单-->
    <dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-web</artifactId>
    </dependency>
    <!-- 测试包,当我们使用 mvn package 的时候该包并不会被打入,因为它的生命周期只在 test 之内-->
    <dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-test</artifactId>
    <scope>test</scope>
    </dependency>

    属性配置

    在 application.properties 文件中分别添加上数据库Mybatis通用MapperPageHelper的属性配置,这里只提供了常见场景的配置,更全的配置可以参考上文所述的文文档(#^.^#)

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    spring.datasource.url=jdbc:mysql://localhost:3306/chapter7?useUnicode=true&characterEncoding=UTF-8&zeroDateTimeBehavior=convertToNull&allowMultiQueries=true&useSSL=false
    spring.datasource.password=root
    spring.datasource.username=root
    # 如果想看到mybatis日志需要做如下配置
    logging.level.com.battcn=DEBUG
    ########## Mybatis 自身配置 ##########
    mybatis.mapper-locations=classpath:com/battcn/mapper/*.xml
    mybatis.type-aliases-package=com.battcn.entity
    # 驼峰命名规范 如:数据库字段是 order_id 那么 实体字段就要写成 orderId
    mybatis.configuration.map-underscore-to-camel-case=true
    ########## 通用Mapper ##########
    # 主键自增回写方法,默认值MYSQL,详细说明请看文档
    mapper.identity=MYSQL
    mapper.mappers=tk.mybatis.mapper.common.BaseMapper
    # 设置 insert 和 update 中,是否判断字符串类型!=''
    mapper.not-empty=true
    # 枚举按简单类型处理
    mapper.enum-as-simple-type=true
    ########## 分页插件 ##########
    pagehelper.helper-dialect=mysql
    pagehelper.params=count=countSql
    pagehelper.reasonable=false
    pagehelper.support-methods-arguments=true

    通用Mapper

    • mapper.enum-as-simple-type: 枚举按简单类型处理,如果有枚举字段则需要加上该配置才会做映射
    • mapper.not-empty: 设置以后,会去判断 insert 和 update 中符串类型!=’’

    分页插件

    • pagehelper.reasonable: 分页合理化参数,默认值为false。当该参数设置为 true 时,pageNum<=0 时会查询第一页, pageNum>pages(超过总数时),会查询最后一页。默认false 时,直接根据参数进行查询。
    • support-methods-arguments: 支持通过 Mapper 接口参数来传递分页参数,默认值false,分页插件会从查询方法的参数值中,自动根据上面 params 配置的字段中取值,查找到合适的值时就会自动分页。

    注意事项

    由于 mybatis.mapper-locations=classpath:com/battcn/mapper/*.xml配置的在java package中,而Spring Boot默认只打入java package -> *.java,所以我们需要给pom.xml文件添加如下内容

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    <build>
    <resources>
    <resource>
    <directory>src/main/resources</directory>
    </resource>
    <resource>
    <directory>src/main/java</directory>
    <includes>
    <include>**/*.xml</include>
    </includes>
    <filtering>true</filtering>
    </resource>
    </resources>
    <plugins>
    <plugin>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-maven-plugin</artifactId>
    </plugin>
    </plugins>
    </build>

    具体编码

    完成基本配置后,接下来进行具体的编码操作。

    表结构

    创建一张 t_user 的表

    1
    2
    3
    4
    5
    6
    CREATE TABLE `t_user` (
    `id` int(8) NOT NULL AUTO_INCREMENT COMMENT '主键自增',
    `username` varchar(50) NOT NULL COMMENT '用户名',
    `password` varchar(50) NOT NULL COMMENT '密码',
    PRIMARY KEY (`id`)
    ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='用户表';

    实体类

    通用Mapper采用了JPA规范包中的注解,这种的设计避免了重复造轮子,更是让Spring Data Jpa的应用可以轻松切换到Mybatis

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    package com.battcn.entity;

    import javax.persistence.GeneratedValue;
    import javax.persistence.GenerationType;
    import javax.persistence.Id;
    import javax.persistence.Table;
    import java.io.Serializable;

    /**
    * @author Levin
    * @since 2018/5/10 0007
    */
    @Table(name = "t_user")
    public class User implements Serializable {

    private static final long serialVersionUID = 8655851615465363473L;

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;
    private String username;
    private String password;
    // TODO 省略get set
    }

    持久层

    为了更好的让熟悉它,此处模拟了一个自定义的SQL,可以发现使用 通用Mapper 后并不会破坏原有代码结构

    UserMapper

    继承 BaseMapper<T> 就可以了,这点是不是有点类似 JpaRepository,同时也可以根据自己需要扩展出更适合自己项目的BaseMapper,它的灵活也是众多开发者喜爱的因素之一

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    package com.battcn.mapper;

    import com.battcn.entity.User;
    import org.apache.ibatis.annotations.Mapper;
    import tk.mybatis.mapper.common.BaseMapper;

    /**
    * t_user 操作,继承 BaseMapper<T> 就可以了,是不是有点类似 JpaRepository
    *
    * @author Levin
    * @since 2018/5/10 0007
    */
    @Mapper
    public interface UserMapper extends BaseMapper<User> {

    /**
    * 根据用户名统计(TODO 假设它是一个很复杂的SQL)
    *
    * @param username 用户名
    * @return 统计结果
    */
    int countByUsername(String username);
    }

    UserMapper 映射文件

    1
    2
    3
    4
    5
    6
    7
    8
    <?xml version="1.0" encoding="UTF-8" ?>
    <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd" >
    <mapper namespace="com.battcn.mapper.UserMapper">

    <select id="countByUsername" resultType="java.lang.Integer">
    SELECT count(1) FROM t_user WHERE username = #{username}
    </select>
    </mapper>

    测试

    完成数据访问层接口后,编写一个junit测试类来检验代码的正确性。

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    package com.battcn;

    import com.battcn.entity.User;
    import com.battcn.mapper.UserMapper;
    import com.github.pagehelper.PageHelper;
    import com.github.pagehelper.PageInfo;
    import org.junit.Test;
    import org.junit.runner.RunWith;
    import org.slf4j.Logger;
    import org.slf4j.LoggerFactory;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.boot.test.context.SpringBootTest;
    import org.springframework.test.context.junit4.SpringRunner;

    import java.util.List;

    /**
    * @author Levin
    */
    @RunWith(SpringRunner.class)
    @SpringBootTest
    public class Chapter7ApplicationTests {

    private static final Logger log = LoggerFactory.getLogger(Chapter7ApplicationTests.class);

    @Autowired
    private UserMapper userMapper;

    @Test
    public void test1() throws Exception {
    final User user1 = new User("u1", "p1");
    final User user2 = new User("u1", "p2");
    final User user3 = new User("u3", "p3");
    userMapper.insertSelective(user1);
    log.info("[user1回写主键] - [{}]", user1.getId());
    userMapper.insertSelective(user2);
    log.info("[user2回写主键] - [{}]", user2.getId());
    userMapper.insertSelective(user3);
    log.info("[user3回写主键] - [{}]", user3.getId());
    final int count = userMapper.countByUsername("u1");
    log.info("[调用自己写的SQL] - [{}]", count);

    // TODO 模拟分页
    userMapper.insertSelective(new User("u1", "p1"));
    userMapper.insertSelective(new User("u1", "p1"));
    userMapper.insertSelective(new User("u1", "p1"));
    userMapper.insertSelective(new User("u1", "p1"));
    userMapper.insertSelective(new User("u1", "p1"));
    userMapper.insertSelective(new User("u1", "p1"));
    userMapper.insertSelective(new User("u1", "p1"));
    userMapper.insertSelective(new User("u1", "p1"));
    userMapper.insertSelective(new User("u1", "p1"));
    userMapper.insertSelective(new User("u1", "p1"));
    // TODO 分页 + 排序 this.userMapper.selectAll() 这一句就是我们需要写的查询,有了这两款插件无缝切换各种数据库
    final PageInfo<Object> pageInfo = PageHelper.startPage(1, 10).setOrderBy("id desc").doSelectPageInfo(() -> this.userMapper.selectAll());
    log.info("[lambda写法] - [分页信息] - [{}]", pageInfo.toString());

    PageHelper.startPage(1, 10).setOrderBy("id desc");
    final PageInfo<User> userPageInfo = new PageInfo<>(this.userMapper.selectAll());
    log.info("[普通写法] - [{}]", userPageInfo);
    }
    }

    总结

    1. Mybatis官方文档: http://www.mybatis.org/mybatis-3/zh/index.html
    2. 通用Mapper文档: https://gitee.com/free/Mapper/wikis/1.1-java?parent=1.integration
    3. 分页插件文档: https://github.com/pagehelper/Mybatis-PageHelper/blob/master/wikis/zh/HowToUse.md
  • 相关阅读:
    旋转坐标转换的矩阵推导
    Union的妙用和注意
    OpenGL环境搭建Windows+Mac+Linux
    RichLabel基于Cocos2dx+Lua v3.x
    Cocos2dx+lua中Color参数的坑
    常见开源协议
    lua序列化(支持循环引用)
    lua标签解析器
    Markdown基本语法
    python3爬取墨迹天气并发送给微信好友,附源码
  • 原文地址:https://www.cnblogs.com/lywJ/p/10715077.html
Copyright © 2011-2022 走看看