Spring Boot统一功能处理深度解析
第一章:为什么需要统一功能处理?
想象你正在开发一个电商系统,包含用户管理、商品管理、订单管理等模块。每个模块都需要:
- 用户身份验证
- 操作日志记录
- 异常统一处理
- 数据格式标准化
如果每个模块都单独实现这些功能:
// 用户模块
@PostMapping("/users")
public User createUser(@RequestBody User user) {try {// 1. 验证token// 2. 记录日志// 3. 业务处理// 4. 统一返回格式} catch (Exception e) {// 5. 异常处理}
}// 商品模块
@PostMapping("/products")
public Product createProduct(@RequestBody Product product) {try {// 重复步骤1-5...} catch (Exception e) {// ...}
}
问题暴露:
- 代码重复率高(DRY原则被破坏)
- 维护困难(修改需多处调整)
- 系统一致性差(不同开发者实现方式不同)
第二章:Spring Boot统一处理四大核心组件
2.1 拦截器(Interceptor) - 请求的"安检门"
作用:在请求到达Controller前/后执行通用逻辑
public class AuthInterceptor implements HandlerInterceptor {@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {// 1. 从请求头获取Token(就像门卫检查你的门禁卡)String token = request.getHeader("Authorization");// 2. 验证Token是否有效(门卫刷卡机验证)if (!validateToken(token)) {// 3. 无效则拒绝进入(设置401状态码)response.setStatus(401);return false; // 中断请求}return true; // 允许进入}// 验证Token的方法(门卫的验证设备)private boolean validateToken(String token) {// 实际验证逻辑(这里简化为非空检查)return token != null && !token.isEmpty();}
}
通俗解释:
- 这个类就像小区的门卫,检查每个进入的人(请求)
preHandle
方法:在业主回家前检查门禁卡(Token)- 如果门禁卡无效(
validateToken
返回false),门卫会拒绝进入(返回401状态码) - 如果有效,就放行(return true)
配置方式:
@Configuration
public class WebConfig implements WebMvcConfigurer {@Overridepublic void addInterceptors(InterceptorRegistry registry) {// 给门卫分配工作区域registry.addInterceptor(new AuthInterceptor()).addPathPatterns("/api/**") // 保护所有/api路径.excludePathPatterns("/api/login"); // 但登录入口不需要检查}
}
通俗解释:
- 这相当于给门卫划定工作范围:
/api/**
:需要检查的所有区域/api/login
:特殊通道(登录入口),不需要检查
2.2 过滤器(Filter) - 请求的"净化器"
作用:处理HTTP请求/响应的原始数据
public class LoggingFilter implements Filter {@Overridepublic void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) {// 1. 记录请求开始时间(水电表开始计数)long startTime = System.currentTimeMillis();// 2. 继续处理请求(让水流/电流继续)chain.doFilter(request, response);// 3. 计算耗时(计算水电用量)long duration = System.currentTimeMillis() - startTime;// 4. 记录日志(生成水电账单)System.out.println("请求耗时: " + duration + "ms");}
}
- 这个过滤器就像水电表:
- 请求开始时记录时间(开水龙头)
chain.doFilter
让请求继续处理(水流)- 请求结束后计算耗时(关水龙头,计算用水量)
- 记录日志(生成水电账单)
与拦截器区别:
特性 | 过滤器(Filter) | 拦截器(Interceptor) |
---|---|---|
作用范围 | Servlet规范 | Spring MVC特有 |
依赖 | 不依赖Spring容器 | 依赖Spring容器 |
获取信息 | 原始HTTP请求/响应 | HandlerMethod信息 |
执行顺序 | 最先执行 | 在DispatcherServlet后执行 |
2.3 切面(AOP) - 功能的"手术刀"
作用:在方法执行前后插入通用逻辑
@Aspect
@Component
public class ServiceMonitorAspect {// 监控Service层方法@Around("execution(* com.example..service.*.*(..))")public Object monitorService(ProceedingJoinPoint pjp) throws Throwable {long start = System.currentTimeMillis();try {return pjp.proceed(); // 执行目标方法} finally {long duration = System.currentTimeMillis() - start;// 记录慢查询if (duration > 500) {log.warn("慢方法: {} 耗时: {}ms", pjp.getSignature(), duration);}}}
}
2.4 控制器增强(@ControllerAdvice) - 全局的"管理员"
作用:统一处理控制器异常和返回格式
@ControllerAdvice
public class GlobalExceptionHandler {// 处理业务异常@ExceptionHandler(BusinessException.class)@ResponseBodypublic ResponseResult<Void> handleBusinessEx(BusinessException e) {return ResponseResult.error(e.getCode(), e.getMessage());}// 处理所有未捕获异常@ExceptionHandler(Exception.class)@ResponseBodypublic ResponseResult<Void> handleException(Exception e) {log.error("系统异常", e);return ResponseResult.error(500, "系统繁忙");}
}
第三章:统一功能处理实战案例
3.1 统一响应格式
// 标准响应体
public class ResponseResult<T> {private int code;private String msg;private T data;// 成功响应public static <T> ResponseResult<T> success(T data) {return new ResponseResult<>(200, "成功", data);}// 错误响应public static <T> ResponseResult<T> error(int code, String msg) {return new ResponseResult<>(code, msg, null);}
}// 控制器统一返回
@RestController
public class UserController {@GetMapping("/users/{id}")public ResponseResult<User> getUser(@PathVariable Long id) {User user = userService.findById(id);return ResponseResult.success(user);}
}
3.2 统一异常处理
@ControllerAdvice // 1. 声明这是全局异常处理器
public class GlobalExceptionHandler {// 2. 处理业务异常(如订单处理失败)@ExceptionHandler(BusinessException.class)@ResponseBodypublic ResponseResult<Void> handleBusinessEx(BusinessException e) {// 返回标准错误格式return ResponseResult.error(e.getCode(), e.getMessage());}// 3. 处理所有未预料异常(系统崩溃)@ExceptionHandler(Exception.class)@ResponseBodypublic ResponseResult<Void> handleException(Exception e) {// 记录错误日志(就像应急系统记录事故)e.printStackTrace();// 返回友好提示return ResponseResult.error(500, "系统繁忙,请稍后再试");}
}
@ControllerAdvice
:这是整个小区的应急中心@ExceptionHandler
:不同类型的应急处理小组BusinessException
:处理业务问题(如订单错误)Exception
:处理所有未预料的问题(系统崩溃)
- 当发生问题时,系统会自动调用对应的方法处理
3.3 统一日志记录
@Aspect
@Component
public class ControllerLogAspect {@Around("@within(org.springframework.web.bind.annotation.RestController)")public Object logController(ProceedingJoinPoint pjp) throws Throwable {// 获取请求信息HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.currentRequestAttributes()).getRequest();// 记录请求日志log.info("请求开始 => URI: {}, 参数: {}", request.getRequestURI(), Arrays.toString(pjp.getArgs()));long start = System.currentTimeMillis();try {Object result = pjp.proceed();// 记录响应日志log.info("请求完成 <= URI: {}, 耗时: {}ms", request.getRequestURI(), System.currentTimeMillis() - start);return result;} catch (Exception e) {// 记录异常日志log.error("请求异常 <= URI: {}, 错误: {}", request.getRequestURI(), e.getMessage());throw e;}}
}
3.4 统一身份认证
public class JwtAuthInterceptor implements HandlerInterceptor {@Autowiredprivate JwtTokenService tokenService;@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {// 1. 获取TokenString token = request.getHeader("Authorization");if (StringUtils.isEmpty(token)) {sendError(response, 401, "未提供认证信息");return false;}// 2. 验证TokenClaims claims = tokenService.parseToken(token);if (claims == null) {sendError(response, 401, "无效的Token");return false;}// 3. 存储用户信息UserContext.setCurrentUser(claims.getSubject());return true;}private void sendError(HttpServletResponse response, int status, String msg) {response.setStatus(status);response.setContentType("application/json");try {response.getWriter().write(new ResponseResult<>(status, msg).toString());} catch (IOException e) {log.error("响应写入失败", e);}}
}
第四章:高级应用场景
4.1 接口限流(Rate Limiting)
@Aspect
@Component
public class RateLimitAspect {private final Map<String, RateLimiter> limiters = new ConcurrentHashMap<>();@Around("@annotation(rateLimit)")public Object rateLimit(ProceedingJoinPoint pjp, RateLimit rateLimit) throws Throwable {String key = getRateLimitKey(pjp);RateLimiter limiter = limiters.computeIfAbsent(key, k -> RateLimiter.create(rateLimit.value()));if (limiter.tryAcquire()) {return pjp.proceed();} else {throw new BusinessException(429, "请求过于频繁");}}private String getRateLimitKey(ProceedingJoinPoint pjp) {// 根据方法+IP生成唯一keyMethodSignature signature = (MethodSignature) pjp.getSignature();HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes()).getRequest();return signature.getMethod().getName() + ":" + request.getRemoteAddr();}
}// 使用注解
@GetMapping("/api/data")
@RateLimit(value = 10) // 每秒10次
public ResponseResult<Data> getData() {// ...
}
4.2 多租户数据隔离
public class TenantInterceptor implements HandlerInterceptor {@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {// 从请求头获取租户IDString tenantId = request.getHeader("X-Tenant-ID");if (StringUtils.isNotBlank(tenantId)) {TenantContext.setCurrentTenant(tenantId);}return true;}
}// 在MyBatis拦截器中应用
@Intercepts({@Signature(type = Executor.class, method = "query", ...)})
public class TenantInterceptor implements Interceptor {@Overridepublic Object intercept(Invocation invocation) throws Throwable {// 自动添加租户过滤条件String tenantId = TenantContext.getCurrentTenant();if (tenantId != null) {MappedStatement ms = (MappedStatement) invocation.getArgs()[0];Object parameter = invocation.getArgs()[1];// 修改SQL添加 tenant_id = #{tenantId}}return invocation.proceed();}
}
4.3 API版本控制
public class ApiVersionInterceptor implements HandlerInterceptor {@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {// 从URL路径获取版本号String path = request.getRequestURI();Matcher matcher = Pattern.compile("/v(\\d+)/").matcher(path);if (matcher.find()) {int version = Integer.parseInt(matcher.group(1));ApiVersionContext.setVersion(version);}return true;}
}// 控制器根据版本路由
@RestController
@RequestMapping("/api/users")
public class UserController {@GetMapping@ApiVersion(1) // v1版本public ResponseResult<List<User>> getUsersV1() {// 旧版逻辑}@GetMapping@ApiVersion(2) // v2版本public ResponseResult<List<UserDto>> getUsersV2() {// 新版逻辑}
}
第五章:性能优化与最佳实践
5.1 组件执行顺序优化
请求进入
Filter-Interceptor-preHandle-Controller-Interceptor-postHandle-Filter-after-Response
优化策略:
- 过滤器优先处理原始数据(如请求体缓存)
- 拦截器处理业务相关逻辑(如认证、日志)
- 切面处理具体方法逻辑(如性能监控)
- @ControllerAdvice最后处理异常和响应
5.2 避免的陷阱
-
循环依赖问题:
-
@Component public class AuthInterceptor implements HandlerInterceptor {@Autowiredprivate AuthService authService; // 可能导致循环依赖 }
解决方案:使用
@Lazy
延迟注入 -
@Lazy @Autowired private AuthService authService;
线程安全问题:
-
public class UserContext {private static final ThreadLocal<User> currentUser = new ThreadLocal<>(); }
性能热点:
-
// 避免在拦截器/过滤器中执行耗时操作 public boolean preHandle(...) {// 错误:执行数据库查询User user = userRepository.findById(userId); }
5.3 最佳实践总结
-
职责分离原则:
- 过滤器:处理原始HTTP请求/响应
- 拦截器:处理业务相关前置/后置逻辑
- 切面:处理具体方法逻辑
- 控制器增强:统一异常和响应处理
-
配置顺序原则:
-
@Override public void addInterceptors(InterceptorRegistry registry) {// 1. 认证拦截器(最先执行)registry.addInterceptor(authInterceptor);// 2. 日志拦截器registry.addInterceptor(logInterceptor);// 3. 其他业务拦截器 }
性能监控指标:
-
// 使用Micrometer监控 @Autowired private MeterRegistry meterRegistry;public void afterCompletion(...) {meterRegistry.counter("api.requests", "uri", request.getRequestURI(),"status", String.valueOf(response.getStatus())).increment(); }
第六章:综合实战 - 电商系统统一处理
6.1 系统架构图
6.2 核心配置代码
// 网关统一配置
@Configuration
public class GatewayConfig {@Beanpublic GlobalFilter customFilter() {return (exchange, chain) -> {// 1. 统一添加请求IDServerHttpRequest request = exchange.getRequest().mutate().header("X-Request-ID", UUID.randomUUID().toString()).build();return chain.filter(exchange.mutate().request(request).build());};}
}// 微服务统一配置
@Configuration
public class ServiceConfig implements WebMvcConfigurer {@Overridepublic void addInterceptors(InterceptorRegistry registry) {registry.addInterceptor(new TenantInterceptor());registry.addInterceptor(new AuthInterceptor());}@Beanpublic FilterRegistrationBean<LoggingFilter> loggingFilter() {FilterRegistrationBean<LoggingFilter> bean = new FilterRegistrationBean<>();bean.setFilter(new LoggingFilter());bean.setOrder(Ordered.HIGHEST_PRECEDENCE);return bean;}
}
6.3 业务服务示例
@RestController
@RequestMapping("/orders")
public class OrderController {@PostMappingpublic ResponseResult<Order> createOrder(@Valid @RequestBody OrderCreateDTO dto) {// 无需处理认证、日志、异常等Order order = orderService.createOrder(dto);return
相关文章:

Spring Boot统一功能处理深度解析
第一章:为什么需要统一功能处理? 想象你正在开发一个电商系统,包含用户管理、商品管理、订单管理等模块。每个模块都需要: 用户身份验证操作日志记录异常统一处理数据格式标准化 如果每个模块都单独实现这些功能: …...

世事无常,比较复杂,人可以简单一点
2025年6月5日日,17~28℃,一般 待办: 宣讲会 职称材料的最后检查 职称材料有错误,需要修改 期末考试试题启用 教学技能大赛PPT 遇见:部门宣传泰国博士项目、硕士项目、本科项目。 感受或反思:东南亚博士…...
使用 Docker Compose 安装 PostgreSQL 16
前面是指南,后面是实际工作日志。 1. 创建 docker-compose.yml 文件 yaml 复制 下载 version: 3.9 services:postgres:image: postgres:16container_name: postgres-16environment:POSTGRES_USER: your_username # 替换为你的用户名POSTGRES_PASSWORD: your…...
每日算法刷题Day23 6.5:leetcode二分答案3道题,用时1h40min(有点慢)
8. 3007.价值和小于等于K的最大数字(中等,学习,太难,先过) 3007. 价值和小于等于 K 的最大数字 - 力扣(LeetCode) 思想 1.给你一个整数 k 和一个整数 x 。整数 num 的价值是它的二进制表示中在 x,2x,3x …...
【Android基础回顾】七:内存管理机制
Android 的内存管理机制是一个多层次的复杂系统,旨在高效利用有限的物理内存(RAM),在保证前台应用流畅运行的同时,尽可能在后台保留更多应用以提高启动速度(多任务)。 它的核心机制结合了 Linu…...

数据结构哈希表总结
349. 两个数组的交集 力扣题目链接(opens new window) 题意:给定两个数组,编写一个函数来计算它们的交集。 说明: 输出结果中的每个元素一定是唯一的。 我们可以不考虑输出结果的顺序。 public int[] intersection(int[] nums1, int[] num…...
Spring事务失效-----十大常见场景及解决方案全解析
Spring事务失效的常见场景及原因分析 Spring事务管理是开发中的核心功能,但在实际应用中可能因各种原因导致事务失效。以下是常见的事务失效场景及详细解析: 1. 方法未被Spring管理 场景:使用new关键字直接创建对象,而非通过Spring容器注入原因:Spring事务基于AOP代理,…...
KMP 算法中 next 数组的构建函数 get_next
KMP 算法中 next 数组的构建函数 get_next ,负责计算模式串的 next 数组,核心是通过递推找到每个位置的 “最长相等前缀后缀长度”。(下标从 1 开始): 一、函数作用 get_next(SString T, int next[]) 的任务…...

IDEA 开发PHP配置调试插件XDebug
1、安装PHP环境 为了方便,使用的PhpStudy。 安装路径:D:\resources\phpstudy_pro\Extensions\php\php7.3.4nts 2、下载Xdebug Xdebug: Downloads 选择对应的版本下载,本次使用的是7.3。 3、配置Xdebug 在php.ini中添加Xdebug配置。 D…...

奇异值分解(SVD):线性代数在AI大模型中的核心工具
🧑 博主简介:CSDN博客专家、CSDN平台优质创作者,高级开发工程师,数学专业,10年以上C/C, C#, Java等多种编程语言开发经验,拥有高级工程师证书;擅长C/C、C#等开发语言,熟悉Java常用开…...
矩阵分解相关知识点总结(二)
文章目录 三、矩阵的QR分解3.1、Givens矩阵与Givens变换3.2、Householder矩阵与Householder变换3.3、QR分解 书接上文矩阵分解相关知识点总结(一) 三、矩阵的QR分解 3.1、Givens矩阵与Givens变换 设非零列向量 x ∈ R n \bm{x}\in {\bf{R}}^n x∈Rn及单…...

MySQL——视图 用户管理 语言访问
目录 视图 用户管理 数据库权限 访问 准备工作 使用函数 mysql界面级工具 连接池 视图 这里的视图与事务中的读视图是两个不同的概念:视图是一个虚拟表,其内容由查询定义。同真实的表一样,视图包含一系列带有名称的列和行数据。视图的…...

二、Sqoop 详细安装部署教程
作者:IvanCodes 日期:2025年6月2日 专栏:Sqoop教程 Apache Sqoop 是一个强大的工具,用于在 Hadoop (HDFS, Hive, HBase) 与关系型数据库 (如 MySQL, PostgreSQL, Oracle) 之间高效传输数据。本教程将详细指导您如何根据官方网站截…...
用Python开启游戏开发之旅
在当今丰富多彩的数字娱乐世界中,游戏以其独特的魅力吸引着无数人的目光。而Python这门功能强大又简洁易懂的编程语言,也为游戏开发打开了一扇充满创意的大门。 一、选择Python的理由 Python之所以备受游戏开发者青睐,有诸多原因。其一&#…...
React 第五十四节 Router中useRevalidator的使用详解及案例分析
前言 useRevalidator 是 React Router v6.4 引入的一个强大钩子,用于在数据路由(Data Router)中手动触发路由数据的重新验证(revalidation)。 它在需要主动刷新数据而不改变路由位置的场景中非常有用。 一、useReval…...

【C语言预处理详解(下)】--#和##运算符,命名约定,命令行定义 ,#undef,条件编译,头文件的包含,嵌套文件包含,其他预处理指令
目录 五.#和##运算符 5.1--#运算符 5.2--##运算符 六.命名约定,#undef,命令行定义 6.1--命名约定 6.2--#undef 6.3--命名行定义 七.条件编译 常见的条件编译指令: 1.普通的条件编译: 2.多个分支的条件编译(可以利用条…...

03.搭建K8S集群
K8S集群搭建的方式 目前主流的搭建k8s集群的方式有kubeadm、minikube、二进制包三种方式: kubeadm(本案例搭建方式) 是一个工具,用于快速搭建kubernetes集群,目前应该是比较方便和推荐的,简单易用 kubea…...

RDMA简介3之四种子协议对比
RDMA协议共有四种子协议,分别为InfiniBand、iWARP、RoCE v1和RoCE v2协议。这四种协议使用统一的RDMA API,但在具体的网络层级实现上有所不同,如图1所示,接下来将分别介绍这四种子协议。 图1 RDMA四种子协议网络层级关系图 Infin…...

【最新版】西陆洗车系统源码全开源+uniapp前端+搭建教程
一.系统介绍 一款基于ThinkPHPUniapp开发的多门店洗车系统,包含用户端(小程序)、门店员工端(小程序)、门店端(PC)、平台管理端(PC)。 门店分连锁门店和独立门店…...
力扣LeetBook数组和字符串--二维数组
1.旋转矩阵 题目链接 想了那么久的各种旋转,对角线,其实把问题搞复杂了。 旋转90度的本质无非就是转置镜像对称 转置是什么?:将矩阵的行和列互换。 镜像对称:把矩阵从中间对折,互换位置 矩阵 A A [ 1 3 0…...

Linux开发工具(apt,vim,gcc)
目录 yum/apt包管理器 Linux编辑器 vim 1.见一见vim 2.vim的多模式 3.命令模式底行模式等 4.vim的配置 Linux编译器 gcc/g 1.预处理(宏替换) 2.编译(生成汇编) 3.汇编(生成机器可识别代码) 4.连…...
C# ExcelWorksheet 贴图
C# ExcelWorksheet 贴图 在C#中,如果你想在Excel工作表中插入图片(例如,在ExcelWorksheet中贴图),你可以使用ClosedXML或EPPlus这样的库来操作Excel文件。下面我将分别介绍如何使用这两个库来实现这一功能。 使用ClosedXML 首先,确保你已经安装了ClosedXML包。你可以通…...

鸿蒙Next开发真机调试签名申请流程
背景: 在学习鸿蒙next开发应用的初期总是会遇到一堆的问题,毕竟鸿蒙next开发不管是他的ArKTS语言还是他的开发工具DevEco Studio都还在起步阶段,就像当初的Android起步一样,总会有资料不足的一些问题。就比如我们学习下载完DevEco…...

[yolov11改进系列]基于yolov11引入上下文锚点注意力CAA的python源码+训练源码
【CAA介绍】 本文记录的是基于CAA注意力模块的RT-DETR目标检测改进方法研究。在远程遥感图像或其他大尺度变化的图像中目标检测任务中,为准确提取其长距离上下文信息,需要解决大目标尺度变化和多样上下文信息时的不足的问题。CAA能够有效捕捉长距离依赖…...
【Elasticsearch】 查询优化方式
在优化Elasticsearch的查询性能时,可以从多个维度着手,包括索引设计、查询优化、集群配置、数据管理以及监控分析等。常见的优化方式和策略有以下几种: 一、索引优化 合理设计字段类型: 字段类型选择: 对于不需要分词的…...
Xcode 16.4 + iOS 18 系统运行时崩溃:___cxa_current_primary_exception 符号丢失的原因与解决方案
在使用 Xcode 16.4 构建项目,运行到 iOS 18.3 或更早版本系统(包括模拟器)时,出现了如下的运行时崩溃: dyld[22183]: Symbol not found: ___cxa_current_primary_exceptionReferenced from: /.../WidgetOn.app/Widget…...

【linux】全志Tina预编译一个so库文件到根文件系统/usr/lib/下
一、sdk中新建文件夹 路径: V:\t113\work3\t113\openwrt\package\feeds\libs\md5util md5util为需要注入的库文件夹。 文件结构 libs md5util files libmd5util.so makefile etc.. 二、编写makefile include $(TOPDIR)/rules.mkPKG_NAME : md5util PKG_VERSIO…...

C# 类和继承(成员访回修饰符)
成员访回修饰符 本章之前的两节阐述了类的可访问性。对类的可访问性,只有两种修饰符:internal和public。 本节阐述成员的可访问性。类的可访问性描述了类的可见性;成员的可访问性描述了类成员的可 见性。 声明在类中的每个成员对系统的不同…...
c++ stl容器之map用法
目录 (1)map介绍 (2)map、multimap、unordered_map区别 (3)map用法 1.map接口表 2.使用举例 插入数据与遍历数据 查找关键字和值 删除元素 按照值排序 (4)multimap用法 &…...

Linux-文件管理及归档压缩
1.根下的目录作用说明: /:Linux系统中所有的文件都在根下/bin:(二进制命令目录)存放常用的用户命令/boot:系统启动时的引导文件(内核的引导配置文件,grub配置文件,内核配置文件) 例…...