多租户实现方案

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

方案 描述 优点 缺点 使用场景
独立数据库 每个租户一个独立的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;
    }
}
相关推荐
我是唐青枫2 分钟前
Java MyBatis 实战指南:XML 映射、动态 SQL 与数据访问层设计
java·mybatis
摇滚侠4 分钟前
Spring 零基础入门到进阶 面向切面 AOP 52-60
java·后端·spring
就改了13 分钟前
微服务接口性能优化:CompletableFuture 并行聚合实践
java·微服务·性能优化
林森lsjs15 分钟前
【日耕一题】4. 较为复杂情况下的求和
java·开发语言
Hui Baby16 分钟前
虚拟线程整理
java
白露与泡影31 分钟前
2026秋招冲刺:1000道Java高频面试题(各大厂考点汇总)
java·开发语言·面试
IT龟苓膏35 分钟前
Java 并发基础:进程、线程、线程状态、synchronized、volatile 一篇讲清
java·开发语言·jvm
weixin_4467291636 分钟前
java中class类没有打进war包中
java
哭哭啼1 小时前
pgSql 事务篇
java·数据库·postgresql
架构源启1 小时前
Spring AI进阶系列(17)- 未来展望与职业发展:Java 工程师迈向 AI 工程化与智能体架构的路线图
java·人工智能·spring