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

MyBatis拦截器黑科技:不修改业务代码实现动态数据权限控制

MyBatis拦截器黑科技零侵入实现企业级数据权限管控在当今企业级应用开发中数据权限控制是一个无法回避的核心需求。传统方案往往需要在每个SQL语句中硬编码权限条件或者通过AOP切面批量修改Mapper接口这些方法要么维护成本高要么灵活性不足。而MyBatis拦截器提供了一种优雅的解决方案——无需修改业务代码通过动态SQL改写实现精细化的数据权限控制。1. MyBatis拦截器机制深度解析MyBatis拦截器本质上基于JDK动态代理和责任链模式允许开发者在SQL执行的关键节点插入自定义逻辑。与Spring AOP不同它直接作用于MyBatis核心组件提供了更底层的控制能力。1.1 四大可拦截组件及其生命周期MyBatis开放了四个核心组件的拦截点构成完整的SQL执行流水线组件类型拦截时机典型应用场景Executorupdate/query/commit等操作执行前后缓存控制、事务管理、SQL执行监控ParameterHandler参数设置阶段参数加密/解密、参数校验ResultSetHandler结果集处理阶段结果集二次加工、敏感数据脱敏StatementHandlerSQL准备和执行阶段SQL重写、分页处理、动态字段控制1.2 拦截器核心API工作原理每个自定义拦截器需要实现Interceptor接口的三个关键方法public interface Interceptor { // 核心拦截逻辑 Object intercept(Invocation invocation) throws Throwable; // 生成代理对象 Object plugin(Object target); // 读取配置参数 void setProperties(Properties properties); }其中plugin方法的标准实现应该始终使用MyBatis提供的工具类Override public Object plugin(Object target) { return Plugin.wrap(target, this); }Plugin.wrap()方法内部会检查目标对象是否符合Intercepts注解定义的拦截条件只有匹配时才会创建代理对象这种设计避免了不必要的代理开销。2. 动态数据权限实战方案2.1 基于ThreadLocal的权限参数传递企业级应用中权限参数通常来自当前用户上下文。我们通过ThreadLocal实现线程安全的参数传递public class DataPermissionContext { private static final ThreadLocalPermissionParam CONTEXT new ThreadLocal(); public static void set(PermissionParam param) { CONTEXT.set(param); } public static PermissionParam get() { return CONTEXT.get(); } public static void clear() { CONTEXT.remove(); } // 权限参数封装类 Data public static class PermissionParam { private Long userId; private Long deptId; private ListLong roleIds; // 其他业务字段... } }在拦截器中可以这样获取权限参数PermissionParam permission DataPermissionContext.get(); if (permission null) { throw new IllegalStateException(数据权限上下文未设置); }2.2 SQL动态改写引擎拦截StatementHandler.prepare方法对原始SQL进行智能改写Intercepts({ Signature(type StatementHandler.class, method prepare, args {Connection.class, Integer.class}) }) public class DataPermissionInterceptor implements Interceptor { Override public Object intercept(Invocation invocation) throws Throwable { StatementHandler handler (StatementHandler) invocation.getTarget(); BoundSql boundSql handler.getBoundSql(); // 获取原始SQL String originalSql boundSql.getSql(); // 解析SQL类型 SqlCommandType sqlType getSqlCommandType(handler); // 动态添加权限条件 String modifiedSql applyDataPermission(originalSql, sqlType); // 通过反射修改SQL Field field boundSql.getClass().getDeclaredField(sql); field.setAccessible(true); field.set(boundSql, modifiedSql); return invocation.proceed(); } private String applyDataPermission(String sql, SqlCommandType sqlType) { PermissionParam permission DataPermissionContext.get(); if (permission null) return sql; // 使用SQL解析器分析语句结构 SQLStatement stmt SQLUtils.parseSingleStatement(sql, JdbcUtils.MYSQL); if (stmt instanceof Select) { return processSelect((Select) stmt, permission); } else if (stmt instanceof Update) { return processUpdate((Update) stmt, permission); } // 其他SQL类型处理... return sql; } }2.3 多租户隔离实现方案对于SaaS系统需要在SQL中自动注入租户隔离条件private String addTenantCondition(Select select, PermissionParam permission) { Table from select.getFrom(); String alias from.getAlias() null ? : from.getAlias() .; // 构造租户条件表达式 BinaryExpression tenantCond new BinaryExpression( alias tenant_id, , String.valueOf(permission.getTenantId()) ); // 合并到WHERE条件 if (select.getWhere() null) { select.setWhere(tenantCond); } else { select.setWhere(new BinaryExpression( select.getWhere(), AND, tenantCond )); } return SQLUtils.toSQLString(select); }3. 高级应用场景与性能优化3.1 行列级权限组合控制实现同时控制可访问的数据行和可见字段public class ColumnPermission { private String tableName; private SetString visibleColumns; private SetString sensitiveColumns; public boolean isColumnAllowed(String column) { return visibleColumns.contains(column.toUpperCase()); } public boolean isSensitiveColumn(String column) { return sensitiveColumns.contains(column.toUpperCase()); } } // 在SQL改写阶段应用列权限 private String applyColumnPermission(Select select, ColumnPermission permission) { select.getSelectList().forEach(item - { if (item instanceof SelectItem) { SelectItem selectItem (SelectItem) item; String column selectItem.getExpr().toString(); if (!permission.isColumnAllowed(column)) { selectItem.setExpr(new SQLIdentifierExpr(NULL)); } else if (permission.isSensitiveColumn(column)) { selectItem.setExpr(new SQLMethodInvokeExpr(MASK, new SQLIdentifierExpr(column))); } } }); return SQLUtils.toSQLString(select); }3.2 性能优化关键点SQL解析缓存使用LRU缓存已解析的SQL语句结构条件短路机制对于超级管理员跳过权限过滤批量操作优化特殊处理批量插入/更新语句动态代理优化减少不必要的代理嵌套// 性能优化后的plugin方法实现 Override public Object plugin(Object target) { if (target instanceof StatementHandler) { return Plugin.wrap(target, this); } return target; // 非目标类型直接返回避免多余代理 }4. 生产环境最佳实践4.1 完整拦截器实现示例Intercepts(Signature(type StatementHandler.class, method prepare, args {Connection.class, Integer.class})) Slf4j public class AdvancedDataPermissionInterceptor implements Interceptor { private final SQLParserFeature[] features { SQLParserFeature.EnableSQLBinaryOpExprGroup, SQLParserFeature.UseInsertColumnsCache }; Override public Object intercept(Invocation invocation) throws Throwable { long start System.currentTimeMillis(); try { StatementHandler handler (StatementHandler) invocation.getTarget(); BoundSql boundSql handler.getBoundSql(); // 跳过无需处理的SQL类型 if (shouldSkip(boundSql)) { return invocation.proceed(); } String originalSql boundSql.getSql(); String modifiedSql processSql(originalSql); if (!originalSql.equals(modifiedSql)) { log.debug(SQL rewritten: {}, modifiedSql); resetBoundSql(boundSql, modifiedSql); } return invocation.proceed(); } finally { log.debug(Permission check cost: {}ms, System.currentTimeMillis() - start); } } private String processSql(String sql) { try { SQLStatement stmt SQLUtils.parseSingleStatement(sql, JdbcUtils.MYSQL, features); PermissionParam permission DataPermissionContext.get(); if (permission null || permission.isAdmin()) { return sql; } if (stmt instanceof Select) { return processSelect((Select) stmt, permission); } else if (stmt instanceof Update) { return processUpdate((Update) stmt, permission); } else if (stmt instanceof Delete) { return processDelete((Delete) stmt, permission); } return sql; } catch (Exception e) { log.warn(SQL parse error, skip permission control, e); return sql; } } // 其他工具方法... }4.2 Spring Boot集成配置Configuration public class MyBatisConfig { Bean public DataPermissionInterceptor dataPermissionInterceptor() { DataPermissionInterceptor interceptor new DataPermissionInterceptor(); // 可配置属性注入 Properties properties new Properties(); properties.setProperty(enableColumnPermission, true); interceptor.setProperties(properties); return interceptor; } Bean public ConfigurationCustomizer configurationCustomizer() { return configuration - { // 确保拦截器在分页插件之后执行 configuration.addInterceptor(dataPermissionInterceptor()); }; } }4.3 常见问题排查指南SQL改写失效检查清单确认拦截器已正确注册到Configuration检查Intercepts注解配置是否正确验证ThreadLocal中权限参数是否设置查看MyBatis日志确认最终执行的SQL性能问题定位方法使用Arthas监控拦截器执行耗时检查是否出现SQL解析瓶颈确认没有不必要的重复代理多数据源适配方案Bean Primary public SqlSessionFactory masterSessionFactory( Qualifier(masterDataSource) DataSource dataSource, DataPermissionInterceptor interceptor) throws Exception { SqlSessionFactoryBean bean new SqlSessionFactoryBean(); bean.setDataSource(dataSource); // 其他配置... bean.setPlugins(new Interceptor[]{interceptor}); return bean.getObject(); }在实际项目中这种非侵入式的数据权限方案相比传统方式减少了80%以上的权限相关代码特别是在快速迭代的业务系统中开发人员可以完全专注于业务逻辑实现而不用担心权限控制的传播问题。

相关文章:

MyBatis拦截器黑科技:不修改业务代码实现动态数据权限控制

MyBatis拦截器黑科技:零侵入实现企业级数据权限管控 在当今企业级应用开发中,数据权限控制是一个无法回避的核心需求。传统方案往往需要在每个SQL语句中硬编码权限条件,或者通过AOP切面批量修改Mapper接口,这些方法要么维护成本高…...

从零搭建QT(C++)开发环境到实战部署YOLOV5模型

1. 环境准备:从零搭建QT开发环境 第一次接触QT开发的朋友可能会被各种安装选项搞懵,我刚开始配置环境时也踩过不少坑。这里分享一个经过验证的安装方案,适用于大多数Linux系统(以Ubuntu为例)。 首先需要安装基础编译工…...

好写作AI:毕业论文的“智能魔法棒”,解锁学术新境界

在学术的征途中,毕业论文如同一座巍峨的山峰,让无数攀登者既期待又畏惧。它不仅是对多年学习成果的检验,更是个人智慧与创造力的集中展现。但面对复杂的结构、严谨的逻辑、浩瀚的文献,以及那令人头疼的格式要求,你是否…...

不止于仿真:用Cadence Virtuoso IC617的Marker和计算器功能高效分析工艺角(以SMIC 0.18um为例)

高效工艺角分析:Cadence Virtuoso IC617的Marker与计算器高阶应用 在集成电路设计领域,工艺角分析是验证设计鲁棒性的关键环节。传统的手动测量方法不仅效率低下,还容易引入人为误差。本文将深入探讨如何利用Cadence Virtuoso IC617中的Advan…...

Codex CLI实战:5分钟搞定React Hooks重构与数据库迁移(附避坑指南)

Codex CLI实战:5分钟搞定React Hooks重构与数据库迁移(附避坑指南) 在快节奏的现代开发中,效率工具的价值愈发凸显。最近半年,身边不少团队开始将Codex CLI作为日常开发的"瑞士军刀"——特别是处理那些重复性…...

Windows Defender系统优化工具:提升系统性能的终极方案

Windows Defender系统优化工具:提升系统性能的终极方案 【免费下载链接】windows-defender-remover A tool which is uses to remove Windows Defender in Windows 8.x, Windows 10 (every version) and Windows 11. 项目地址: https://gitcode.com/gh_mirrors/wi…...

别再纠结选哪个了!手把手教你根据项目需求选对Go框架:Gin、Kratos还是Zero?

实战指南:如何为你的Go项目精准匹配框架——Gin、Kratos与Zero深度解析 当启动一个新项目时,选择正确的框架往往决定了后续开发的顺畅程度。面对Gin、Kratos和Zero这三个主流Go框架,很多开发者会陷入选择困难。本文将带你从实际项目需求出发&…...

告别乱码黑屏:FBTFT驱动ST7789屏幕的常见问题排查与修复指南

告别乱码黑屏:FBTFT驱动ST7789屏幕的常见问题排查与修复指南 当你在树莓派或香橙派上尝试用FBTFT驱动ST7789屏幕时,最令人沮丧的莫过于接好线后——屏幕要么一片漆黑,要么疯狂闪烁乱码。作为一款被移入Linux内核staging目录的驱动框架&#x…...

告别手动计算!用EB工具链高效配置S32K144的Dio与Port模块

告别手动计算!用EB工具链高效配置S32K144的Dio与Port模块 在汽车电子开发中,S32K1XX系列MCU因其出色的实时性和可靠性成为主流选择。但面对数百个引脚配置,传统手动计算PCR值、逐项填写寄存器的方式不仅效率低下,还容易引入人为错…...

OpenClaw+Phi-3-vision无障碍应用:图片转语音助手的实现

OpenClawPhi-3-vision无障碍应用:图片转语音助手的实现 1. 项目背景与动机 去年夏天,我在社区图书馆做志愿者时遇到一位视障读者。他需要将纸质书籍内容转换成语音,但现有工具要么操作复杂,要么需要付费订阅。这件事让我开始思考…...

性价比高的南昌实体店线上获客哪个靠谱

在南昌,实体店想要在竞争激烈的市场中脱颖而出,线上获客是关键。然而,面对众多的线上获客途径,哪个才靠谱且性价比高呢?今天,我们就来详细探讨一下,同时为大家推荐南昌琨瑜象限本地生活运营服务…...

Balena Etcher在Arch Linux上的终极安装指南:3种简单方法轻松搞定镜像烧录

Balena Etcher在Arch Linux上的终极安装指南:3种简单方法轻松搞定镜像烧录 【免费下载链接】etcher Flash OS images to SD cards & USB drives, safely and easily. 项目地址: https://gitcode.com/GitHub_Trending/et/etcher Balena Etcher是一款安全易…...

OpenClaw安装 Skill 完整指南:从哪里找、怎么安装到怎么验证

OpenClaw安装 Skill 完整指南:从哪里找、怎么安装到怎么验证 关键词:OpenClaw、OpenClaw Skill、OpenClaw安装Skill、OpenClaw教程、AI智能体、EasyClaw 摘要:很多人开始接触 OpenClaw 后,很快就会遇到一个问题:Skil…...

是德N5771A直流电源/keysight N5771A

是德N5771A直流电源/keysight N5771A 是德N5771A 探头是一款 直流电源 ,主要特点包括‌: ‌输出额定值‌:电压为300伏,电流为5安培,功率为1500瓦‌ ‌接口标准‌:支持 GPIB 、 LAN 、 USB 接口&#xff0…...

CATIA 转 SolidWorks 高效转换技巧:迪威模型网实战解析

1. CATIA与SolidWorks转换的必要性 在工程设计领域,CATIA和SolidWorks就像两个说着不同方言的工程师。我见过太多团队因为文件格式不通用而耽误进度,特别是当汽车供应商收到主机厂的CATIA文件时,经常需要熬夜加班做格式转换。迪威模型网的在线…...

从PID到阻抗:机器人柔顺控制的模型演进与动力学角色

1. PID控制的本质与局限性 我第一次接触机器人控制时,导师就让我从PID开始学起。这个诞生于上世纪的控制算法,至今仍是工业界的"万金油"。但真正用它做过机器人项目的人都知道,PID就像一把锤子——简单粗暴但缺乏灵活性。 PID的核心…...

打字不如说话,说话不如截图——AI 代码助手的多模态输入实践偈

整体排查思路 我们的目标是验证以下三个环节是否正常: 登录成功时:服务器是否正确生成了Session并返回了包含正确 JSESSIONID的Cookie给浏览器。 浏览器端:浏览器是否成功接收并存储了该Cookie。 后续请求:浏览器在执行查询等操作…...

[特殊字符] 《网络知识和Servlet重点知识整理》

一、网络作用(基础认知) 核心作用:实现不同设备之间的数据传输与通信,支撑互联网应用(网页、APP、游戏、视频等)。 信息传递:客户端 ↔ 服务器 资源共享:文件、数据库、计算资源 分…...

YOLOv12解决方案实战:智能安防、交通监控、工业检测三大场景应用

YOLOv12解决方案实战:智能安防、交通监控、工业检测三大场景应用 【免费下载链接】yolov12 [NeurIPS 2025] YOLOv12: Attention-Centric Real-Time Object Detectors 项目地址: https://gitcode.com/gh_mirrors/yo/yolov12 YOLOv12作为NeurIPS 2025最新发布的…...

避坑指南:在实现LL(1)语法分析器时,SELECT集合计算的那些‘坑’与调试技巧

LL(1)语法分析器实战:SELECT集合计算的七大陷阱与可视化调试方法论 当你按照教科书实现了一个LL(1)语法分析器,却发现它错误地将已知的LL(1)文法判定为非LL(1)文法时,问题往往出在SELECT集合的计算逻辑上。本文将揭示开发者常踩的七个关键陷阱…...

3步掌握Adobe-GenP:开源工具助力创意工作流效率提升

3步掌握Adobe-GenP:开源工具助力创意工作流效率提升 【免费下载链接】Adobe-GenP Adobe CC 2019/2020/2021/2022/2023 GenP Universal Patch 3.0 项目地址: https://gitcode.com/gh_mirrors/ad/Adobe-GenP 在数字创意领域,Adobe Creative Cloud套…...

旋转编码器底层驱动库:轻量级正交解码与抗抖动设计

1. 旋转编码器底层驱动库技术解析与工程实践旋转编码器(Rotary Encoder)是嵌入式系统中最为基础且高频使用的机电输入设备之一,广泛应用于工业HMI、电机调速面板、音频设备音量调节、医疗设备参数设定等场景。其核心价值在于提供无触点、高寿…...

别再只传明文了!SpringBoot若依框架接口Base64加解密避坑指南

若依框架接口安全升级:Base64编码传输的实战陷阱与解决方案 在前后端分离架构中,数据安全传输一直是开发者关注的焦点。最近接手一个金融类项目改造,客户明确要求所有接口数据必须经过编码处理。当我信心满满地准备用Base64方案快速实现时&am…...

告别“权限不足”:手把手教你用CobaltStrike的Bypass UAC功能搞定Windows提权

实战指南:利用CobaltStrike突破Windows权限限制 当你手握一个普通用户权限的Beacon会话,却卡在"请求的操作需要提升"的提示前,这种挫败感每个渗透测试员都深有体会。Windows的用户账户控制(UAC)就像一堵无形的墙,将普通…...

千问3.5-9B提示工程:提升OpenClaw复杂任务分解能力

千问3.5-9B提示工程:提升OpenClaw复杂任务分解能力 1. 为什么需要优化任务拆解能力 上周我让OpenClaw执行"整理上季度销售数据并邮件发送给团队"时,AI直接把原始CSV文件作为附件群发——这显然不是人类想要的"整理"结果。这个尴尬…...

ESPS USB MSC 调试全过程记录酪

背景 在软件开发的漫长旅途中,"构建"这个词往往让人又爱又恨。爱的是,一键点击,代码变成产品,那是程序员最迷人的时刻;恨的是,维护那一堆乱糟糟的构建脚本,简直是噩梦。 在很多项目中…...

Win11共享打印机报错0x00000709?别慌,试试这个注册表一键修复脚本

Win11共享打印机0x00000709错误终极修复指南:注册表脚本与深度解析 遇到Win11共享打印机报错0x00000709时,很多用户会陷入反复重装驱动、重启打印服务的循环中。这个看似简单的网络打印故障,实则与Windows的RPC通信协议配置密切相关。本文将提…...

Verdi 快速上手:信号追踪与波形调试实战

1. Verdi工具入门:数字IC调试的瑞士军刀 刚接触数字IC设计时,最让我头疼的就是仿真波形调试。密密麻麻的信号线像一团乱麻,根本不知道从哪里下手。直到同事推荐了Verdi,这个被业界称为"调试神器"的工具彻底改变了我的工…...

在Laravel 8中配置和使用基于IP的API限流策略

引言在Web开发中,API限流是保护服务器免受恶意请求和滥用的重要手段。Laravel框架提供了简单而强大的限流功能,可以轻松实现基于IP地址的请求限制。本文将详细介绍如何在Laravel 8中配置和使用基于IP的API限流策略。为什么需要API限流?API限流…...

AirPlay协议开源实现全攻略:从Raspberry Pi到Linux的5种方案实测

AirPlay协议开源实现全攻略:从Raspberry Pi到Linux的5种方案实测 在智能家居和多媒体共享领域,AirPlay协议因其出色的用户体验和苹果生态的广泛普及而备受关注。然而,官方AirPlay服务仅限于苹果自家设备,这促使开发者社区涌现出多…...