百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT技术 > 正文

MyBatis实战三之使用MyBatis-Plus动态表分表

wptr33 2024-12-15 17:13 21 浏览

前言

随着我们开发的阅读平台内容越来越丰富,注册的用户越来越多,单表的数据存储结构已经不能满足项目的需求。我们需要对用户表进行必要的拆分以提高系统的可靠性和可用性,目前我们把用户表拆分为 256个子表,如图:

对数据表拆分的方式有很多,经典的如 Sharding JDBC,这里我介绍如何使用MyBatis-Plus的动态表名插件 实现数据的拆分。

动态表名插件是什么?

DynamicTableNameInnerInterceptor是mybatis-plug的一个拦截器插件,可以自己定义需要拦截的表单,然后对它进行加工,这时mybatis-plus就会把SQL代码的表名加上你的这个装饰。

  • 原理为解析替换设定表名为处理器的返回表名,表名建议可以定义复杂一些避免误替换
  • 例如:真实表名为 user 设定为 UserTableNameHandler 处理器替换为 user_0001 等
  • TableNameHandler tableNameHandler是一个接口。使用动态表名插件时,必须要有 TableNameHandler的实现类。
public interface TableNameHandler {
    String dynamicTableName(String sql, String tableName);
}

DynamicTableNameInnerInterceptor 的changeTable 方法 如下

栗子

模拟使用场景

一个 User 对应多张表(多张表结构一致,只有表名称不同),在使用时,可以动态映射表名称。
比如:按照用户ID分表,根据自定义的Hash算法找到我们需要的表名。

表user_0001

CREATE TABLE `user_0001` (
  `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT COMMENT '主键ID',
  `nick_name` varchar(32) NOT NULL COMMENT '用户昵称',
  `gender` tinyint(2) unsigned NOT NULL COMMENT '性别',
  `type` tinyint(2) unsigned NOT NULL COMMENT '用户类型',
  `is_vip` tinyint(2) unsigned NOT NULL COMMENT '是否VIP',
  `grade` tinyint(2) unsigned NOT NULL COMMENT '等级',
  `create_time` datetime NOT NULL COMMENT '创建时间',
  `address` varchar(256) DEFAULT NULL,
  PRIMARY KEY (`id`),
  KEY `idx_nickname` (`nick_name`)
) ENGINE=InnoDB AUTO_INCREMENT=11 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci COMMENT='用户表';

表user_0002

CREATE TABLE `user_0002` (
  `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT COMMENT '主键ID',
  `nick_name` varchar(32) NOT NULL COMMENT '用户昵称',
  `gender` tinyint(2) unsigned NOT NULL COMMENT '性别',
  `type` tinyint(2) unsigned NOT NULL COMMENT '用户类型',
  `is_vip` tinyint(2) unsigned NOT NULL COMMENT '是否VIP',
  `grade` tinyint(2) unsigned NOT NULL COMMENT '等级',
  `create_time` datetime NOT NULL COMMENT '创建时间',
  `address` varchar(256) DEFAULT NULL,
  PRIMARY KEY (`id`),
  KEY `idx_nickname` (`nick_name`)
) ENGINE=InnoDB AUTO_INCREMENT=11 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci COMMENT='用户表';

引入依赖

<dependencies>
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
        </dependency>
        <dependency>
            <groupId>cn.hutool</groupId>
            <artifactId>hutool-all</artifactId>
            <version>5.8.3</version>
        </dependency>
        <dependency>
            <groupId>cn.hutool</groupId>
            <artifactId>hutool-core</artifactId>
            <version>5.7.13</version>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <version>1.18.24</version>
        </dependency>
        <!--mysql-->
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>8.0.23</version>
        </dependency>
        <!--druid-->
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>druid</artifactId>
            <version>1.2.11</version>
        </dependency>
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>druid-spring-boot-starter</artifactId>
            <version>1.1.21</version>
        </dependency>
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>fastjson</artifactId>
            <type>jar</type>
            <version>2.0.32</version>
        </dependency>
        <dependency>
            <groupId>com.baomidou</groupId>
            <artifactId>mybatis-plus-boot-starter</artifactId>
            <version>3.4.2</version>
        </dependency>
        <dependency>
            <groupId>com.baomidou</groupId>
            <artifactId>dynamic-datasource-spring-boot-starter</artifactId>
            <version>3.4.1</version>
        </dependency>
    </dependencies>

DbTableUtil 分表类

/**
 * Hash 分表
 * @author yangyanping
 * @date 2024-03-29
 */
public class DbTableUtil {
    private static final int TABLE_SIZE = 256;

    public DbTableUtil() {
    }

    public static String getTableIndex(Object shardKey, int sliceSize) {
        int hashValue = Math.abs(shardKey.hashCode());
        int tableIndex = hashValue % sliceSize + 1;
        return String.format("%04d", tableIndex);
    }

    public static String hash(long key, int sliceSize) {
        long hashValue = Math.abs(key);
        int tableIndex = (int)(hashValue % (long)sliceSize + 1L);
        return String.format("%04d", tableIndex);
    }

    public static String hash(long key) {
        return hash(key, 256);
    }
}

请求参数动态表名传递辅助类

/**
 * 请求参数动态表名传递辅助类
 *
 * @author yangyanping
 * @date 2024-03-29
 */
public class RequestDynamicTableNameHelper {

    /**
     * 请求参数存取(表名)。请求参数自定义,官方Demo定义为ThreadLocal<Map<String, Object>>
     */
    private static final ThreadLocal<String> REQUEST_DATA = new ThreadLocal<>();

    /**
     * 设置请求参数
     *
     * @param requestData
     *            请求参数-表名
     */
    public static void setRequestData(String requestData) {
        REQUEST_DATA.set(requestData);
    }

    /**
     * 获取请求参数
     *
     * @return 请求参数-表名
     */
    public static String getRequestData() {
        return REQUEST_DATA.get();
    }

    /**
     * 移除获取请求参数(表名)
     */
    public static void remove() {
        REQUEST_DATA.remove();
    }
}

动态表名处理器

/**
 * 动态表名处理器---根据用户ID分表
 * mybatis-plus提供了动态表名处理器接口TableNameHandler,
 * 只需要在系统中实现该接口,并作为插件加载到mybatis-plus中就可以使用
 *
 * @author yangyanping
 * @date 2023-02-15
 */
@Slf4j
public class UserTableNameHandler implements TableNameHandler {
    @Override
    public String dynamicTableName(String sql, String tableName) {
        //表名增加hash值
        String userId = RequestDynamicTableNameHelper.getRequestData();
        String tabSuffix = tableName + "_" + DbTableUtil.getTableIndex(Convert.toLong(userId), 256);
        log.info("UserAscribe#tabSuffix={}", tabSuffix);

        return tabSuffix;
    }
}

注入动态表名插件

MyBatisPlusConfig配置中添加动态表名 DynamicTableNameInnerInterceptor插件。

使用多个功能插件时注意顺序关系,官方建议使用如下顺序:

  • 多租户,动态表名
  • 分页,乐观锁
  • sql 性能规范,防止全表更新与删除
@Configuration
@MapperScan("com.mysql.dao.user")
public class MyBatisPlusConfig {

    @Bean
    public MybatisPlusInterceptor mybatisPlusInterceptor() {
        MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
        interceptor.addInnerInterceptor(dynamicTableNameInnerInterceptor());

        return interceptor;
    }

    @Bean
    public DynamicTableNameInnerInterceptor dynamicTableNameInnerInterceptor(){
        DynamicTableNameInnerInterceptor dynamicTableNameInnerInterceptor =
                new DynamicTableNameInnerInterceptor();

        Map<String, TableNameHandler> tableNameHandlerMap = new HashMap<>();

        TableNameHandler userTableNameHandler = new UserTableNameHandler();
        tableNameHandlerMap.put("user", userTableNameHandler);

        dynamicTableNameInnerInterceptor.setTableNameHandlerMap(tableNameHandlerMap);

        return dynamicTableNameInnerInterceptor;
    }
}

UserInfo 类和 UserInfoDao

@Data
@TableName("user")
public class UserInfo {
    private Long id;

    private String nickName;

    private Integer gender;

    private Integer type;

    private Integer isVip;

    private Integer grade;

    private String address;
}
public interface BookInfoDao {
}

单元测试

@RunWith(SpringRunner.class)
@SpringBootTest(classes = WebApplication.class)
public class DynamicTableNameTest {

    @Resource
    private UserInfoDao userInfoDao;

    @Test
    public void insert0001() {
        RequestDynamicTableNameHelper.setRequestData("1");
        UserInfo userInfo = new UserInfo();
        userInfo.setId(1L);
        userInfo.setNickName("user1");
        userInfo.setGender(1);
        userInfo.setType(1);
        userInfo.setGrade(1);
        userInfo.setIsVip(1);
        userInfo.setAddress("");
        userInfo.setCreateTime(new Date());
        userInfoDao.insert(userInfo);
    }

    @Test
    public void insert0256() {
        RequestDynamicTableNameHelper.setRequestData("256");
        UserInfo userInfo = new UserInfo();
        userInfo.setId(256L);
        userInfo.setNickName("user256");
        userInfo.setGender(1);
        userInfo.setType(1);
        userInfo.setGrade(1);
        userInfo.setIsVip(1);
        userInfo.setAddress("");
        userInfo.setCreateTime(new Date());
        userInfoDao.insert(userInfo);
    }
}

相关推荐

【推荐】一款开源免费、美观实用的后台管理系统模版

如果您对源码&技术感兴趣,请点赞+收藏+转发+关注,大家的支持是我分享最大的动力!!!项目介绍...

Android架构组件-App架构指南,你还不收藏嘛

本指南适用于那些已经拥有开发Android应用基础知识的开发人员,现在想了解能够开发出更加健壮、优质的应用程序架构。首先需要说明的是:AndroidArchitectureComponents翻...

高德地图经纬度坐标批量拾取(高德地图批量查询经纬度)

使用方法在桌面上新建一个index.txt文件,把下面的代码复制进去保存,再把文件名改成index.html保存,双击运行打开即可...

flutter系列之:UI layout简介(flutter ui设计)

简介对于一个前端框架来说,除了各个组件之外,最重要的就是将这些组件进行连接的布局了。布局的英文名叫做layout,就是用来描述如何将组件进行摆放的一个约束。...

Android开发基础入门(一):UI与基础控件

Android基础入门前言:...

iOS的布局体系-流式布局MyFlowLayout

iOS布局体系的概览在我的CSDN博客中的几篇文章分别介绍MyLayout布局体系中的视图从一个方向依次排列的线性布局(MyLinearLayout)、视图层叠且停靠于父布局视图某个位置的框架布局(M...

TDesign企业级开源设计系统越发成熟稳定,支持 Vue3 / 小程序

TDesing发展越来越好了,出了好几套组件库,很成熟稳定了,新项目完全可以考虑使用。...

WinForm实现窗体自适应缩放(winform窗口缩放)

众所周知,...

winform项目——仿QQ即时通讯程序03:搭建登录界面

上两篇文章已经对CIM仿QQ即时通讯项目进行了需求分析和数据库设计。winform项目——仿QQ即时通讯程序01:原理及项目分析...

App自动化测试|原生app元素定位方法

元素定位方法介绍及应用Appium方法定位原生app元素...

61.C# TableLayoutPanel控件(c# tabcontrol)

摘要TableLayoutPanel在网格中排列内容,提供类似于HTML元素的功能。TableLayoutPanel控件允许你将控件放在网格布局中,而无需精确指定每个控件的位置。其单元格...

想要深入学习Android性能优化?看完这篇直接让你一步到位

...

12个python数据处理常用内置函数(python 的内置函数)

在python数据分析中,经常需要对字符串进行各种处理,例如拼接字符串、检索字符串等。下面我将对python中常用的内置字符串操作函数进行介绍。1.计算字符串的长度-len()函数str1='我爱py...

如何用Python程序将几十个PDF文件合并成一个PDF?其实只要这四步

假定你有一个很无聊的任务,需要将几十个PDF文件合并成一个PDF文件。每一个文件都有一个封面作为第一页,但你不希望合并后的文件中重复出现这些封面。即使有许多免费的程序可以合并PDF,很多也只是简单的将...

Python入门知识点总结,Python三大数据类型、数据结构、控制流

Python基础的重要性不言而喻,是每一个入门Python学习者所必备的知识点,作为Python入门,这部分知识点显得很庞杂,内容分支很多,大部分同学在刚刚学习时一头雾水。...