Spring Boot统一功能处理深度解析

news2025/7/26 20:09:53

第一章:为什么需要统一功能处理?

想象你正在开发一个电商系统,包含用户管理、商品管理、订单管理等模块。每个模块都需要:

  1. 用户身份验证
  2. 操作日志记录
  3. 异常统一处理
  4. 数据格式标准化

如果每个模块都单独实现这些功能:

// 用户模块
@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 {
    @Override
    public 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 {
    @Override
    public 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 {
    @Override
    public 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");
    }
}
  • 这个过滤器就像水电表:
    1. 请求开始时记录时间(开水龙头)
    2. chain.doFilter让请求继续处理(水流)
    3. 请求结束后计算耗时(关水龙头,计算用水量)
    4. 记录日志(生成水电账单)

与拦截器区别

特性过滤器(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)
    @ResponseBody
    public ResponseResult<Void> handleBusinessEx(BusinessException e) {
        return ResponseResult.error(e.getCode(), e.getMessage());
    }
    
    // 处理所有未捕获异常
    @ExceptionHandler(Exception.class)
    @ResponseBody
    public 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)
    @ResponseBody
    public ResponseResult<Void> handleBusinessEx(BusinessException e) {
        // 返回标准错误格式
        return ResponseResult.error(e.getCode(), e.getMessage());
    }
    
    // 3. 处理所有未预料异常(系统崩溃)
    @ExceptionHandler(Exception.class)
    @ResponseBody
    public 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 {
    
    @Autowired
    private JwtTokenService tokenService;
    
    @Override
    public boolean preHandle(HttpServletRequest request, 
                             HttpServletResponse response, 
                             Object handler) {
        // 1. 获取Token
        String token = request.getHeader("Authorization");
        if (StringUtils.isEmpty(token)) {
            sendError(response, 401, "未提供认证信息");
            return false;
        }
        
        // 2. 验证Token
        Claims 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生成唯一key
        MethodSignature 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 {
    
    @Override
    public boolean preHandle(HttpServletRequest request, 
                             HttpServletResponse response, 
                             Object handler) {
        // 从请求头获取租户ID
        String 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 {
    @Override
    public 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 {
    
    @Override
    public 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

优化策略

  1. 过滤器优先处理原始数据(如请求体缓存)
  2. 拦截器处理业务相关逻辑(如认证、日志)
  3. 切面处理具体方法逻辑(如性能监控)
  4. @ControllerAdvice最后处理异常和响应
5.2 避免的陷阱
  1. 循环依赖问题

  2. @Component
    public class AuthInterceptor implements HandlerInterceptor {
        @Autowired
        private AuthService authService; // 可能导致循环依赖
    }
    

    解决方案:使用@Lazy延迟注入

  3. @Lazy
    @Autowired
    private AuthService authService;
    

    线程安全问题

  4. public class UserContext {
        private static final ThreadLocal<User> currentUser = new ThreadLocal<>();
    }
    

    性能热点

  5. // 避免在拦截器/过滤器中执行耗时操作
    public boolean preHandle(...) {
        // 错误:执行数据库查询
        User user = userRepository.findById(userId);
    }
    
    5.3 最佳实践总结
  6. 职责分离原则

    • 过滤器:处理原始HTTP请求/响应
    • 拦截器:处理业务相关前置/后置逻辑
    • 切面:处理具体方法逻辑
    • 控制器增强:统一异常和响应处理
  7. 配置顺序原则

  8. @Override
    public void addInterceptors(InterceptorRegistry registry) {
        // 1. 认证拦截器(最先执行)
        registry.addInterceptor(authInterceptor);
        
        // 2. 日志拦截器
        registry.addInterceptor(logInterceptor);
        
        // 3. 其他业务拦截器
    }
    

    性能监控指标

  9. // 使用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 {
    
    @Bean
    public GlobalFilter customFilter() {
        return (exchange, chain) -> {
            // 1. 统一添加请求ID
            ServerHttpRequest 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 {
    
    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(new TenantInterceptor());
        registry.addInterceptor(new AuthInterceptor());
    }
    
    @Bean
    public 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 {
    
    @PostMapping
    public ResponseResult<Order> createOrder(@Valid @RequestBody OrderCreateDTO dto) {
        // 无需处理认证、日志、异常等
        Order order = orderService.createOrder(dto);
        return

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2402081.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

世事无常,比较复杂,人可以简单一点

2025年6月5日日&#xff0c;17~28℃&#xff0c;一般 待办&#xff1a; 宣讲会 职称材料的最后检查 职称材料有错误&#xff0c;需要修改 期末考试试题启用 教学技能大赛PPT 遇见&#xff1a;部门宣传泰国博士项目、硕士项目、本科项目。 感受或反思&#xff1a;东南亚博士…

数据结构哈希表总结

349. 两个数组的交集 力扣题目链接(opens new window) 题意&#xff1a;给定两个数组&#xff0c;编写一个函数来计算它们的交集。 说明&#xff1a; 输出结果中的每个元素一定是唯一的。 我们可以不考虑输出结果的顺序。 public int[] intersection(int[] nums1, int[] num…

IDEA 开发PHP配置调试插件XDebug

1、安装PHP环境 为了方便&#xff0c;使用的PhpStudy。 安装路径&#xff1a;D:\resources\phpstudy_pro\Extensions\php\php7.3.4nts 2、下载Xdebug Xdebug: Downloads 选择对应的版本下载&#xff0c;本次使用的是7.3。 3、配置Xdebug 在php.ini中添加Xdebug配置。 D…

奇异值分解(SVD):线性代数在AI大模型中的核心工具

&#x1f9d1; 博主简介&#xff1a;CSDN博客专家、CSDN平台优质创作者&#xff0c;高级开发工程师&#xff0c;数学专业&#xff0c;10年以上C/C, C#, Java等多种编程语言开发经验&#xff0c;拥有高级工程师证书&#xff1b;擅长C/C、C#等开发语言&#xff0c;熟悉Java常用开…

MySQL——视图 用户管理 语言访问

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

二、Sqoop 详细安装部署教程

作者&#xff1a;IvanCodes 日期&#xff1a;2025年6月2日 专栏&#xff1a;Sqoop教程 Apache Sqoop 是一个强大的工具&#xff0c;用于在 Hadoop (HDFS, Hive, HBase) 与关系型数据库 (如 MySQL, PostgreSQL, Oracle) 之间高效传输数据。本教程将详细指导您如何根据官方网站截…

【C语言预处理详解(下)】--#和##运算符,命名约定,命令行定义 ,#undef,条件编译,头文件的包含,嵌套文件包含,其他预处理指令

目录 五.#和##运算符 5.1--#运算符 5.2--##运算符 六.命名约定&#xff0c;#undef&#xff0c;命令行定义 6.1--命名约定 6.2--#undef 6.3--命名行定义 七.条件编译 常见的条件编译指令&#xff1a; 1.普通的条件编译&#xff1a; 2.多个分支的条件编译(可以利用条…

03.搭建K8S集群

K8S集群搭建的方式 目前主流的搭建k8s集群的方式有kubeadm、minikube、二进制包三种方式&#xff1a; kubeadm&#xff08;本案例搭建方式&#xff09; 是一个工具&#xff0c;用于快速搭建kubernetes集群&#xff0c;目前应该是比较方便和推荐的&#xff0c;简单易用 kubea…

RDMA简介3之四种子协议对比

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

【最新版】西陆洗车系统源码全开源+uniapp前端+搭建教程

一.系统介绍 一款基于ThinkPHPUniapp开发的多门店洗车系统&#xff0c;包含用户端&#xff08;小程序&#xff09;、门店员工端&#xff08;小程序&#xff09;、门店端&#xff08;PC&#xff09;、平台管理端&#xff08;PC&#xff09;。 门店分连锁门店和独立门店&#xf…

Linux开发工具(apt,vim,gcc)

目录 yum/apt包管理器 Linux编辑器 vim 1.见一见vim 2.vim的多模式 3.命令模式底行模式等 4.vim的配置 Linux编译器 gcc/g 1.预处理&#xff08;宏替换&#xff09; 2.编译&#xff08;生成汇编&#xff09; 3.汇编&#xff08;生成机器可识别代码&#xff09; 4.连…

鸿蒙Next开发真机调试签名申请流程

背景&#xff1a; 在学习鸿蒙next开发应用的初期总是会遇到一堆的问题&#xff0c;毕竟鸿蒙next开发不管是他的ArKTS语言还是他的开发工具DevEco Studio都还在起步阶段&#xff0c;就像当初的Android起步一样&#xff0c;总会有资料不足的一些问题。就比如我们学习下载完DevEco…

[yolov11改进系列]基于yolov11引入上下文锚点注意力CAA的python源码+训练源码

【CAA介绍】 本文记录的是基于CAA注意力模块的RT-DETR目标检测改进方法研究。在远程遥感图像或其他大尺度变化的图像中目标检测任务中&#xff0c;为准确提取其长距离上下文信息&#xff0c;需要解决大目标尺度变化和多样上下文信息时的不足的问题。CAA能够有效捕捉长距离依赖…

【linux】全志Tina预编译一个so库文件到根文件系统/usr/lib/下

一、sdk中新建文件夹 路径&#xff1a; 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# 类和继承(成员访回修饰符)

成员访回修饰符 本章之前的两节阐述了类的可访问性。对类的可访问性&#xff0c;只有两种修饰符&#xff1a;internal和public。 本节阐述成员的可访问性。类的可访问性描述了类的可见性&#xff1b;成员的可访问性描述了类成员的可 见性。 声明在类中的每个成员对系统的不同…

Linux-文件管理及归档压缩

1.根下的目录作用说明&#xff1a; /&#xff1a;Linux系统中所有的文件都在根下/bin&#xff1a;(二进制命令目录)存放常用的用户命令/boot&#xff1a;系统启动时的引导文件&#xff08;内核的引导配置文件&#xff0c;grub配置文件&#xff0c;内核配置文件&#xff09; 例…

微软认证考试科目众多?该如何选择?

在云计算、人工智能、数据分析等技术快速发展的今天&#xff0c;微软认证&#xff08;Microsoft Certification&#xff09;已成为IT从业者、开发者、数据分析师提升竞争力的重要凭证。但面对众多考试科目&#xff0c;很多人不知道如何选择。本文将详细介绍微软认证的考试方向、…

Dify工作流实践—根据word需求文档编写测试用例到Excel中

前言 这篇文章依赖到的操作可查阅我之前的文章&#xff1a; dify里的大模型是怎么添加进来的&#xff1a;在Windows本地部署Dify详细操作 flask 框架app.route()函数的开发和调用&#xff1a;PythonWeb开发框架—Flask工程创建和app.route使用详解 结构化提示词的编写&…

【LC实战派】小智固件编译

这篇写给立创吴总&#xff0c;是节前答应他配合git代码的说明&#xff1b;也给所有对小智感兴趣的小伙伴。 请多提意见&#xff0c;让这份文档更有价值 - 第一当然是拉取源码 - git clone https://github.com/78/xiaozhi-esp32.git 完成后&#xff0c;先查看固件中实际的…

jdbcTemplate.query备忘录

jdbcTemplate.query中使用全部字符串和参数注入&#xff0c; 查询速度为什么差距这么大 如何正确使用JdbcTemplate参数化查询 1、使用?占位符 String sql "SELECT * FROM users WHERE name LIKE ?"; List<User> users jdbcTemplate.query(sql,new Object[…