当前位置: 首页 > news >正文

如何添加“默认给Sql查询语句加上租户条件”的功能

从零实现“默认给 SQL 查询语句加上租户条件”的功能,本质上是利用MyBatis Plus的插件机制配合ThreadLocal上下文来实现的。

我们需要构建一条完整的“数据 -> 规则 -> 执行”的链路。以下是标准化的5 步实现指南


第一步:准备“背包” (定义上下文容器)

你需要一个地方在当前线程中存储“当前租户是谁”。

代码核心:利用ThreadLocal

publicclassTenantContextHolder{// 存放当前租户IDprivatestaticfinalThreadLocal<Long>TENANT_ID=newTransmittableThreadLocal<>();publicstaticvoidsetTenantId(LongtenantId){TENANT_ID.set(tenantId);}publicstaticLonggetTenantId(){returnTENANT_ID.get();}publicstaticvoidclear(){TENANT_ID.remove();}}

第二步:制定“规则” (实现 Handler 接口)

你需要告诉 MyBatis Plus 具体的过滤逻辑:租户ID是多少?列名叫什么?哪些表不需要加?

代码核心:实现TenantLineHandler接口。

@ComponentpublicclassMyTenantLineHandlerimplementsTenantLineHandler{// 1. 告诉 MP,当前租户ID是多少 (从背包里拿)@OverridepublicExpressiongetTenantId(){LongtenantId=TenantContextHolder.getTenantId();// 如果没拿到ID(比如没登录),返回 NullValue 可能会导致报错或查不到数据// 通常这里会做判空或者返回默认值returnnewLongValue(tenantId);}// 2. 告诉 MP,数据库里租户列的名字叫什么@OverridepublicStringgetTenantIdColumn(){return"tenant_id";}// 3. 告诉 MP,哪些表需要忽略 (白名单)@OverridepublicbooleanignoreTable(StringtableName){// A. 全局白名单:系统表(字典、菜单)不需要隔离if("sys_dict".equals(tableName)||"sys_menu".equals(tableName)){returntrue;}// B. 动态白名单:配合 @TenantIgnore 注解使用if(TenantContextHolder.isIgnore()){returntrue;}// 默认:必须加过滤条件returnfalse;}}

第三步:组装“引擎” (配置 MyBatis 拦截器)

有了规则(Handler),你需要把它交给执行者(Interceptor),并把执行者放入 MyBatis 的插件链中。

代码核心:配置MybatisPlusInterceptorBean。

@ConfigurationpublicclassMybatisConfig{@BeanpublicMybatisPlusInterceptormybatisPlusInterceptor(MyTenantLineHandlertenantLineHandler){MybatisPlusInterceptorinterceptor=newMybatisPlusInterceptor();// 核心动作:创建租户拦截器,并注入上面的规则 Handler// ⚠️注意:建议放在分页插件之前interceptor.addInnerInterceptor(newTenantLineInnerInterceptor(tenantLineHandler));// 添加分页插件interceptor.addInnerInterceptor(newPaginationInnerInterceptor(DbType.MYSQL));returninterceptor;}}

第四步:数据“注入” (配置 Web 过滤器)

引擎装好了,但还得有人把燃料(租户ID)塞进第一步的“背包”里。通常是在请求刚进来时处理。

代码核心:实现FilterHandlerInterceptor

@ComponentpublicclassTenantContextFilterimplementsFilter{@OverridepublicvoiddoFilter(ServletRequestrequest,ServletResponseresponse,FilterChainchain){try{// 1. 从请求头 Header 获取 tenant-idHttpServletRequestreq=(HttpServletRequest)request;StringtenantIdStr=req.getHeader("tenant-id");if(tenantIdStr!=null){// 2. 塞进 ThreadLocal 背包TenantContextHolder.setTenantId(Long.valueOf(tenantIdStr));}// 3. 放行,执行后续业务逻辑 (Service -> Mapper -> SQL拦截器)chain.doFilter(request,response);}finally{// 4. 【重要】请求结束,清空背包,防止线程污染TenantContextHolder.clear();}}}

🌟 总结:这一套下来发生了什么?

  1. 请求进来:Filter 从 Header 拿到tenant_id=1,存入TenantContextHolder
  2. 业务查询:你写了userMapper.selectList(null)
  3. 拦截改写TenantLineInnerInterceptor拦截 SQL,调用MyTenantLineHandler
  4. 读取规则:Handler 从TenantContextHolder拿到1
  5. SQL 变身:SQL 被自动拼接为SELECT * FROM user WHERE tenant_id = 1
  6. 请求结束:Filter 清空TenantContextHolder

这就是实现“全自动多租户隔离”的完整标准流程。

http://www.cnnetsun.cn/news/98689.html

相关文章:

  • 1.2 人工智能的多维度定义:弱AI、强AI与超级AI的理论边界
  • 26、量子计算与高维空间探索
  • 基于EmotiVoice的游戏角色语音定制方案设计
  • 15、应对 OWASP 十大安全风险的实用指南
  • LobeChat可用性99.9%保障措施
  • TAFAS:面向非平稳时间序列的测试时自适应预测
  • Dubbo服务提供者失效踢出机制揭秘:原理与实战解析
  • 79、由于提供的内容仅“以下”二字,没有具体信息,无法按照要求生成博客,请你提供更详细的英文内容。
  • 80、由于没有提供第80章的具体英文内容,我无法为你完成博客创作,请你提供英文内容,以便我按照要求输出博客。
  • EmotiVoice能否用于电话自动应答系统?稳定性验证
  • 基于多智能体强化学习的轨道追逃博弈方法
  • 23、Git操作与日志查看全解析
  • 27、Git工作流与分支约定详解
  • EmotiVoice语音合成在品牌语音形象塑造中的战略意义
  • 32、优化你的 Git 使用体验
  • 企业级高校宣讲会管理系统管理系统源码|SpringBoot+Vue+MyBatis架构+MySQL数据库【完整版】
  • 29、深入解析SNMP MIB实现与操作
  • 27、深入探究Bash调试器:功能、结构与操作详解
  • 35、《Bash Shell 技术全解析:从基础到高级应用》
  • 7、Puppet基础设施搭建与配置指南
  • EmotiVoice在语音导航系统中的路径提示优化
  • 47、磁盘存储系统的全面解析与性能优化
  • 54、计算机系统安全与程序威胁深度解析
  • 58、Linux系统:架构、模块与进程管理解析
  • 59、Linux系统调度与内存管理详解
  • 67、操作系统技术解析:从线程存储到系统特性
  • EmotiVoice + GPU算力:实现毫秒级高保真语音生成
  • 基于EmotiVoice的有声内容创作指南:提升听众沉浸感
  • 26、Go 语言并发模式与反射机制详解
  • 30、编程知识综合解析