Mybatis-Plus同时实现分表和表内多租户模式

2023-12-15 00:36

本文主要是介绍Mybatis-Plus同时实现分表和表内多租户模式,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在之前经理的某家公司中,经历了一个saas服务的某些功能的数据量不断变大的过程,因为各种功能和性能的原因想到的方法就是直接按saas租户做分库和按租户对某些数据量大的表做分表。但是在我离职之前这两种方式都未能实现。不过,最近刚好看到Mybatis-Plus的多租户的拦截器功能,想到可以用来做第二种方案的问题的解决方法,因此来尝试一番。 使用最新版Mybatis-Plus

        <dependency>
            <groupId>com.baomidou</groupId>
            <artifactId>mybatis-plus-boot-starter</artifactId>
            <version>3.5.4.1</version>
        </dependency>

需要配合使用DynamicTableNameInnerInterceptor和TenantLineInnerInterceptor

DynamicTableNameInnerInterceptor

利用DynamicTableNameInnerInterceptor主要是用来对某些数据量大的表做分表查询的,这个拦截器可以在执行sql语句的时候动态的修改查询的表名。使用方法如下

   //这里我将租户id写死了。真实的实现中应当从当前登录的数据中获取
   private static final String tenant_id = "zhao";
    @Bean
    public MybatisPlusInterceptor mybatisPlusInterceptor(){
        MybatisPlusInterceptor interceptor =new MybatisPlusInterceptor();
        DynamicTableNameInnerInterceptor dynamicTableNameInnerInterceptor= new DynamicTableNameInnerInterceptor();
        TableNameHandler tableNameHandler = new TableNameHandler() {
            @Override
            public String dynamicTableName(String sql, String tableName) {
                //tenantProperty中动态标识的是哪些表是分表的表,就在哪些分表的表添加租户的表后缀
                //可以将tenantProperty的配置修改为数据库配置也可以,改动更灵活
                if (tenantProperty.getShardingTables().contains(tableName)){
                    return tableName+"_"+tenant_id;
                }
                return tableName;
            }
        };
        dynamicTableNameInnerInterceptor.setTableNameHandler(tableNameHandler);
        interceptor.addInnerInterceptor(dynamicTableNameInnerInterceptor);

        return interceptor;
    }

TenantLineInnerInterceptor

Mybatis自带的无自定义的租户的拦截器会在所有的sql后面加上对应的租户条件,但是我们可以自定义对应的处理租户信息相关的Handler.

public class MultiTenantLineHandler implements TenantLineHandler {

    private TenantProperty tenantProperty;

    public MultiTenantLineHandler(TenantProperty tenantProperty){
        this.tenantProperty =tenantProperty;
    }
    @Override
    public Expression getTenantId() {
        //此处直接使用给定租户,实际实现从登录信息中取出
        return new StringValue("zhao");
    }

    @Override
    public String getTenantIdColumn() {
        //租户列名
        return tenantProperty.getTenantColumn();
    }

    //需要忽略的表的配置
    @Override
    public boolean ignoreTable(String tableName) {
        List<String> ignoreTables = tenantProperty.getIgnoreTables();
        if (ignoreTables.contains(tableName)){
            return true;
        }
        return false;
    }
//    不处理的非租户列的insert
//    @Override
//    public boolean ignoreInsert(List<Column> columns, String tenantIdColumn) {
//        return TenantLineHandler.super.ignoreInsert(columns, tenantIdColumn);
//    }
}

自定义配置类TenantProperty和拦截器整合

配置类

@Data
@Configuration
@ConfigurationProperties(prefix = "tenant")
public class TenantProperty {

    private Boolean enable =true;

    private String tenantColumn="tenant_id";

    private List<String> ignoreTables;

    private List<String> shardingTables;



}

配置信息

tenant:
  enabletrue
  ignoreTables:
    - sharding
  shardingTables:
    - sharding

整合拦截器

@Configuration
@MapperScan("com.zhao.sbsc17.dao")
public class TableTenantConfig {
//    @Bean
//    public MybatisPlusInterceptor mybatisPlusInterceptor(){
//        MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
//        interceptor.addInnerInterceptor(new OptimisticLockerInnerInterceptor());
//        return interceptor;
//    }
    private static final String tenant_id = "zhao";
    @Autowired
    TenantProperty tenantProperty;
    @Bean
    public MybatisPlusInterceptor mybatisPlusInterceptor(){
        MybatisPlusInterceptor interceptor =new MybatisPlusInterceptor();
        interceptor.addInnerInterceptor(new OptimisticLockerInnerInterceptor());
        interceptor.addInnerInterceptor(new TenantLineInnerInterceptor(new MultiTenantLineHandler(tenantProperty)));
        DynamicTableNameInnerInterceptor dynamicTableNameInnerInterceptor= new DynamicTableNameInnerInterceptor();
        TableNameHandler tableNameHandler = new TableNameHandler() {
            @Override
            public String dynamicTableName(String sql, String tableName) {
                if (tenantProperty.getShardingTables().contains(tableName)){
                    return tableName+"_"+tenant_id;
                }
                return tableName;
            }
        };
        dynamicTableNameInnerInterceptor.setTableNameHandler(tableNameHandler);
        interceptor.addInnerInterceptor(dynamicTableNameInnerInterceptor);

        return interceptor;
    }
}

需要说明的是两个拦截器添加的顺序如果不同会有不同的效果,那么也就需要做对应的处理。当前demo我是达到了刚好我要使用的效果。

效果

新建表tenant和sharding_zhao

查询

@RestController
@RequestMapping("tenant")
public class TenantController {

    @Autowired
    ShardingMapper shardingMapper;
    @Autowired
    TenantMapper tenantMapper;

    @GetMapping("tenant")
    public Tenant master(){
        return tenantMapper.selectById(1L);
    }

    @GetMapping("sharding")
    public Sharding sharding(){
        return shardingMapper.selectById(1L);
    }

}

Creating a new SqlSession
SqlSession [org.apache.ibatis.session.defaults.DefaultSqlSession@213354a5] was not registered for synchronization because synchronization is not active
JDBC Connection [com.alibaba.druid.proxy.jdbc.ConnectionProxyImpl@70191485] will not be managed by Spring
==>  Preparing: SELECT id, goods_id, goods_name, num, version, tenant_id FROM tenant WHERE id = ? AND tenant_id = 'zhao'
==> Parameters: 1(Long)
<==    Columns: id, goods_id, goods_name, num, version, tenant_id
<==        Row: 1, 1, 测试master, 1, 1, zhao
<==      Total: 1
Closing non transactional SqlSession [org.apache.ibatis.session.defaults.DefaultSqlSession@213354a5]
Creating a new SqlSession
SqlSession [org.apache.ibatis.session.defaults.DefaultSqlSession@7aebce73] was not registered for synchronization because synchronization is not active
JDBC Connection [com.alibaba.druid.proxy.jdbc.ConnectionProxyImpl@70191485] will not be managed by Spring
==>  Preparing: SELECT id, goods_id, goods_name, num, version FROM sharding_zhao WHERE id = ?
==> Parameters: 1(Long)
<==    Columns: id, goods_id, goods_name, num, version
<==        Row: 1, 1, 1, 1, 1
<==      Total: 1
Closing non transactional SqlSession [org.apache.ibatis.session.defaults.DefaultSqlSession@7aebce73]

新增

Creating a new SqlSession
SqlSession [org.apache.ibatis.session.defaults.DefaultSqlSession@14a9086f] was not registered for synchronization because synchronization is not active
JDBC Connection [com.alibaba.druid.proxy.jdbc.ConnectionProxyImpl@48087898] will not be managed by Spring
==>  Preparing: INSERT INTO sharding_zhao (id, goods_name, num, version) VALUES (?, ?, ?, ?)
==> Parameters: 2(Long), 测试(String), 1(Long), 1(Integer)
<==    Updates: 1
Closing non transactional SqlSession [org.apache.ibatis.session.defaults.DefaultSqlSession@14a9086f]
Creating a new SqlSession
SqlSession [org.apache.ibatis.session.defaults.DefaultSqlSession@53a2e535] was not registered for synchronization because synchronization is not active
JDBC Connection [com.alibaba.druid.proxy.jdbc.ConnectionProxyImpl@2054630b] will not be managed by Spring
==>  Preparing: INSERT INTO tenant (id, goods_name, num, version, tenant_id) VALUES (?, ?, ?, ?, 'zhao')
==> Parameters: 2(Long), 测试(String), 1(Long), 1(Integer)
<==    Updates: 1
Closing non transactional SqlSession [org.apache.ibatis.session.defaults.DefaultSqlSession@53a2e535]

本文由 mdnice 多平台发布

这篇关于Mybatis-Plus同时实现分表和表内多租户模式的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/494472

相关文章

HTML5 getUserMedia API网页录音实现指南示例小结

《HTML5getUserMediaAPI网页录音实现指南示例小结》本教程将指导你如何利用这一API,结合WebAudioAPI,实现网页录音功能,从获取音频流到处理和保存录音,整个过程将逐步... 目录1. html5 getUserMedia API简介1.1 API概念与历史1.2 功能与优势1.3

Java实现删除文件中的指定内容

《Java实现删除文件中的指定内容》在日常开发中,经常需要对文本文件进行批量处理,其中,删除文件中指定内容是最常见的需求之一,下面我们就来看看如何使用java实现删除文件中的指定内容吧... 目录1. 项目背景详细介绍2. 项目需求详细介绍2.1 功能需求2.2 非功能需求3. 相关技术详细介绍3.1 Ja

使用Python和OpenCV库实现实时颜色识别系统

《使用Python和OpenCV库实现实时颜色识别系统》:本文主要介绍使用Python和OpenCV库实现的实时颜色识别系统,这个系统能够通过摄像头捕捉视频流,并在视频中指定区域内识别主要颜色(红... 目录一、引言二、系统概述三、代码解析1. 导入库2. 颜色识别函数3. 主程序循环四、HSV色彩空间详解

PostgreSQL中MVCC 机制的实现

《PostgreSQL中MVCC机制的实现》本文主要介绍了PostgreSQL中MVCC机制的实现,通过多版本数据存储、快照隔离和事务ID管理实现高并发读写,具有一定的参考价值,感兴趣的可以了解一下... 目录一 MVCC 基本原理python1.1 MVCC 核心概念1.2 与传统锁机制对比二 Postg

SpringBoot整合Flowable实现工作流的详细流程

《SpringBoot整合Flowable实现工作流的详细流程》Flowable是一个使用Java编写的轻量级业务流程引擎,Flowable流程引擎可用于部署BPMN2.0流程定义,创建这些流程定义的... 目录1、流程引擎介绍2、创建项目3、画流程图4、开发接口4.1 Java 类梳理4.2 查看流程图4

C++中零拷贝的多种实现方式

《C++中零拷贝的多种实现方式》本文主要介绍了C++中零拷贝的实现示例,旨在在减少数据在内存中的不必要复制,从而提高程序性能、降低内存使用并减少CPU消耗,零拷贝技术通过多种方式实现,下面就来了解一下... 目录一、C++中零拷贝技术的核心概念二、std::string_view 简介三、std::stri

C++高效内存池实现减少动态分配开销的解决方案

《C++高效内存池实现减少动态分配开销的解决方案》C++动态内存分配存在系统调用开销、碎片化和锁竞争等性能问题,内存池通过预分配、分块管理和缓存复用解决这些问题,下面就来了解一下... 目录一、C++内存分配的性能挑战二、内存池技术的核心原理三、主流内存池实现:TCMalloc与Jemalloc1. TCM

OpenCV实现实时颜色检测的示例

《OpenCV实现实时颜色检测的示例》本文主要介绍了OpenCV实现实时颜色检测的示例,通过HSV色彩空间转换和色调范围判断实现红黄绿蓝颜色检测,包含视频捕捉、区域标记、颜色分析等功能,具有一定的参考... 目录一、引言二、系统概述三、代码解析1. 导入库2. 颜色识别函数3. 主程序循环四、HSV色彩空间

Redis Cluster模式配置

《RedisCluster模式配置》:本文主要介绍RedisCluster模式配置,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录分片 一、分片的本质与核心价值二、分片实现方案对比 ‌三、分片算法详解1. ‌范围分片(顺序分片)‌2. ‌哈希分片3. ‌虚

Python实现精准提取 PDF中的文本,表格与图片

《Python实现精准提取PDF中的文本,表格与图片》在实际的系统开发中,处理PDF文件不仅限于读取整页文本,还有提取文档中的表格数据,图片或特定区域的内容,下面我们来看看如何使用Python实... 目录安装 python 库提取 PDF 文本内容:获取整页文本与指定区域内容获取页面上的所有文本内容获取