多租户实现方案

实现多租户的话,一般有以下几种实现方案,如下所示:

方案 描述 优点 缺点 使用场景
独立数据库 每个租户一个独立的DB实例 隔离性最好,物理隔离,单租户故障不影响其他人,备份回复容易 成本最高(连接池管理,迁移脚本执行),运维复杂,资源利用率低 对数据敏感的大客户
独立schema 同一个DB实例,每个租户一个Schema 隔离性较好,逻辑上分开,运维成本适中 跨租户统计困难,数据库连接数有上限,MySQL下Schema过多会影响性能 中型SaaS,对隔离有一定要求,且租户数量可控
共享表+租户id 所有租户共用一套表,通过tenant_id 字段区分 成本最低,资源利用率最高,开发维护简单,跨租户查询容易 隔离性最弱,代码层面一旦漏掉租户id,容易造成数据泄露 绝大多数saas平台,小微租户多,对成本敏感

我的决策:我会优选采用方案三(共享表+租户ID),因为性价比最高。

技术落地:

一、租户上下文的传递

在网关或者拦截器中,从Header中解析出tarentId,并使用ThreadLocal将TarentId放入当前线程的上下文中;

二、数据隔离的核心

利用Mybatis的拦截器,TenantInterceptor实现他的getTenantId方法

多线程情况下,上下文丢失,导致找不到tenantId;

可以利用spring的TaskDecorator,它允许我们在任务执行前后执行制定逻辑,我们可以通过配置,在任务执行前 设置tarentId,在任务执行后清理

bash 复制代码
import org.springframework.context.annotation.Bean;
import org.springframework.core.task.TaskDecorator;
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;

public class ExecutorConfig {

    // 假设这是你用于存储上下文的 ThreadLocal
    private static final ThreadLocal<String> MY_CONTEXT = new ThreadLocal<>();

    @Bean
    public ThreadPoolTaskExecutor taskExecutor() {
        ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
        executor.setCorePoolSize(10);
        executor.setMaxPoolSize(20);
        executor.setQueueCapacity(100);
        
        // 关键配置:设置 TaskDecorator
        executor.setTaskDecorator(new TaskDecorator() {
            @Override
            public Runnable decorate(Runnable runnable) {
                // 1. 在主线程中捕获上下文数据
                String contextValue = MY_CONTEXT.get();
                
                // 2. 返回一个包装后的 Runnable
                return () -> {
                    try {
                        // 3. 在子线程中设置捕获到的上下文
                        if (contextValue != null) {
                            MY_CONTEXT.set(contextValue);
                        }
                        // 4. 执行原始任务
                        runnable.run();
                    } finally {
                        // 5. 任务执行完毕,清理上下文,防止内存泄漏和数据污染
                        MY_CONTEXT.remove();
                    }
                };
            }
        });
        
        executor.initialize();
        return executor;
    }
}
相关推荐
Dicky-_-zhang3 小时前
消息队列Kafka/RocketMQ选型与高可用架构:从单体到100万TPS的演进
java·jvm
晨曦中的暮雨3 小时前
4.15腾讯 CSIG云服务产线 一面
java·开发语言
fake_ss1983 小时前
AI时代学习全栈项目开发的新范式
java·人工智能·学习·架构·个人开发·学习方法
茉莉玫瑰花茶4 小时前
工作流的常见模式 [ 1 ]
java·服务器·前端
未若君雅裁4 小时前
Spring AOP、日志切面与声明式事务原理
java·后端·spring
No8g攻城狮4 小时前
【人大金仓】wsl2+ubuntu22.04安装人大金仓数据库V9
java·数据库·spring boot·非关系型数据库
xiaoerbuyu12335 小时前
开源Java 邮箱 基于SpringBoot+Vue前后端分离的电子邮件
java·开发语言
C+++Python5 小时前
C++ 进阶学习完整指南
java·c++·学习
zhangjw346 小时前
第11篇:Java Map集合详解,HashMap底层原理、哈希冲突、JDK1.8优化、遍历方式彻底吃透
java·开发语言·哈希算法